From c7d907bdc9264a7c599d2167f72f769d57b617fb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Vo=C5=99=C3=AD=C5=A1ek?= Date: Sat, 20 Apr 2024 21:40:03 +0200 Subject: [PATCH 1/5] Use composer InstalledRepository to determine install dir --- src/ExtensionInstaller.php | 42 ++++++++++++++++++++++++++++---------- 1 file changed, 31 insertions(+), 11 deletions(-) diff --git a/src/ExtensionInstaller.php b/src/ExtensionInstaller.php index 33fff66..12bdead 100644 --- a/src/ExtensionInstaller.php +++ b/src/ExtensionInstaller.php @@ -5,34 +5,48 @@ use Composer\Installer\LibraryInstaller; use Composer\Package\PackageInterface; use Composer\Package\Version\VersionParser; +use Composer\Repository\InstalledRepository; use Composer\Repository\InstalledRepositoryInterface; +use Composer\Repository\RootPackageRepository; use Composer\Util\Filesystem; use Composer\Util\ProcessExecutor; use React\Promise\PromiseInterface; abstract class ExtensionInstaller extends LibraryInstaller { + private $roundcubemailInstallPath; + protected $composer_type; - protected function getRoundcubemailInstallPath(): string + protected function setRoundcubemailInstallPath(InstalledRepositoryInterface $installedRepo): void { - $rootPackage = $this->composer->getPackage(); - if ($rootPackage->getName() === 'roundcube/roundcubemail') { + // https://github.com/composer/composer/discussions/11927#discussioncomment-9116893 + $rootPackage = clone $this->composer->getPackage(); + $installedRepo = new InstalledRepository([ + $installedRepo, + new RootPackageRepository($rootPackage), + ]); + + $roundcubemailPackages = $installedRepo->findPackagesWithReplacersAndProviders('roundcube/roundcubemail'); + assert(count($roundcubemailPackages) === 1); + $roundcubemailPackage = $roundcubemailPackages[0]; + + if ($roundcubemailPackage === $rootPackage) { // $this->getInstallPath($package) does not work for root package $this->initializeVendorDir(); - - return dirname($this->vendorDir); + $this->roundcubemailInstallPath = dirname($this->vendorDir); + } else { + $this->roundcubemailInstallPath = $this->getInstallPath($roundcubemailPackage); } + } - $roundcubemailPackage = $this->composer - ->getRepositoryManager() - ->findPackage('roundcube/roundcubemail', '*'); - - return $this->getInstallPath($roundcubemailPackage); + protected function getRoundcubemailInstallPath(): string + { + return $this->roundcubemailInstallPath; } public function getInstallPath(PackageInterface $package) { - if (!$this->supports($package->getType())) { + if (!$this->supports($package->getType()) || $this->roundcubemailInstallPath === null /* install path is not known at download phase */) { return parent::getInstallPath($package); } @@ -43,6 +57,8 @@ public function getInstallPath(PackageInterface $package) public function install(InstalledRepositoryInterface $repo, PackageInterface $package) { + $this->setRoundcubemailInstallPath($repo); + // initialize Roundcube environment if (!defined('INSTALL_PATH')) { define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); @@ -113,6 +129,8 @@ public function install(InstalledRepositoryInterface $repo, PackageInterface $pa public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target) { + $this->setRoundcubemailInstallPath($repo); + // initialize Roundcube environment if (!defined('INSTALL_PATH')) { define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); @@ -186,6 +204,8 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini public function uninstall(InstalledRepositoryInterface $repo, PackageInterface $package) { + $this->setRoundcubemailInstallPath($repo); + // initialize Roundcube environment if (!defined('INSTALL_PATH')) { define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); From c98d6e4a68cf64cafba0e2f9d7f4d435964cec57 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Vo=C5=99=C3=AD=C5=A1ek?= Date: Sun, 21 Apr 2024 01:18:29 +0200 Subject: [PATCH 2/5] Dedup install path computation --- src/ExtensionInstaller.php | 23 ++++++++++++++--------- 1 file changed, 14 insertions(+), 9 deletions(-) diff --git a/src/ExtensionInstaller.php b/src/ExtensionInstaller.php index 12bdead..c0ebdcf 100644 --- a/src/ExtensionInstaller.php +++ b/src/ExtensionInstaller.php @@ -14,8 +14,10 @@ abstract class ExtensionInstaller extends LibraryInstaller { + /** @var string|null */ private $roundcubemailInstallPath; + /** @var string */ protected $composer_type; protected function setRoundcubemailInstallPath(InstalledRepositoryInterface $installedRepo): void @@ -46,13 +48,17 @@ protected function getRoundcubemailInstallPath(): string public function getInstallPath(PackageInterface $package) { - if (!$this->supports($package->getType()) || $this->roundcubemailInstallPath === null /* install path is not known at download phase */) { + if ( + !$this->supports($package->getType()) + || $this->roundcubemailInstallPath === null // install path is not known at download phase + ) { return parent::getInstallPath($package); } $vendorDir = $this->getVendorDir(); - return sprintf('%s/%s', $vendorDir, $this->getPackageName($package)); + return $vendorDir . \DIRECTORY_SEPARATOR + . str_replace('/', \DIRECTORY_SEPARATOR, $this->getPackageName($package)); } public function install(InstalledRepositoryInterface $repo, PackageInterface $package) @@ -70,7 +76,7 @@ public function install(InstalledRepositoryInterface $repo, PackageInterface $pa $postInstall = function () use ($package) { $config_file = $this->rcubeConfigFile(); $package_name = $this->getPackageName($package); - $package_dir = $this->getVendorDir() . \DIRECTORY_SEPARATOR . $package_name; + $package_dir = $this->getInstallPath($package); $extra = $package->getExtra(); if (is_writable($config_file) && \PHP_SAPI === 'cli' && $this->confirmInstall($package_name)) { @@ -143,8 +149,7 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini $fs = new Filesystem(); // backup persistent files e.g. config.inc.php - $package_name = $this->getPackageName($initial); - $package_dir = $this->getVendorDir() . \DIRECTORY_SEPARATOR . $package_name; + $package_dir = $this->getInstallPath($initial); $temp_dir = $package_dir . '-' . sprintf('%010d%010d', mt_rand(), mt_rand()); // make a backup of existing files (for restoring persistent files) @@ -152,7 +157,7 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini $postUpdate = function () use ($target, $extra, $fs, $temp_dir) { $package_name = $this->getPackageName($target); - $package_dir = $this->getVendorDir() . \DIRECTORY_SEPARATOR . $package_name; + $package_dir = $this->getInstallPath($target); // restore persistent files $persistent_files = !empty($extra['roundcube']['persistent-files']) ? $extra['roundcube']['persistent-files'] : ['config.inc.php']; @@ -217,7 +222,7 @@ public function uninstall(InstalledRepositoryInterface $repo, PackageInterface $ $postUninstall = function () use ($package, $config) { // post-uninstall: deactivate package $package_name = $this->getPackageName($package); - $package_dir = $this->getVendorDir() . \DIRECTORY_SEPARATOR . $package_name; + $package_dir = $this->getInstallPath($package); $this->rcubeAlterConfig($package_name, false); @@ -400,7 +405,7 @@ private function rcubeRunScript($script, PackageInterface $package) { $package_name = $this->getPackageName($package); $package_type = $package->getType(); - $package_dir = $this->getVendorDir() . \DIRECTORY_SEPARATOR . $package_name; + $package_dir = $this->getInstallPath($package); // check for executable shell script if (($scriptfile = realpath($package_dir . \DIRECTORY_SEPARATOR . $script)) && is_executable($scriptfile)) { @@ -424,7 +429,7 @@ private function rcubeRunScript($script, PackageInterface $package) } /** - * normalize Roundcube version string. + * Normalize Roundcube version string. */ private static function versionNormalize(string $version): string { From 8c74725e37e50a6f3b60bf1b5c379ec72d9f0774 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Vo=C5=99=C3=AD=C5=A1ek?= Date: Sun, 21 Apr 2024 01:17:32 +0200 Subject: [PATCH 3/5] Improve ternary CS --- src/ExtensionInstaller.php | 4 +++- src/PluginInstaller.php | 8 ++++++-- src/SkinInstaller.php | 8 ++++++-- 3 files changed, 15 insertions(+), 5 deletions(-) diff --git a/src/ExtensionInstaller.php b/src/ExtensionInstaller.php index c0ebdcf..ac3cb2e 100644 --- a/src/ExtensionInstaller.php +++ b/src/ExtensionInstaller.php @@ -160,7 +160,9 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini $package_dir = $this->getInstallPath($target); // restore persistent files - $persistent_files = !empty($extra['roundcube']['persistent-files']) ? $extra['roundcube']['persistent-files'] : ['config.inc.php']; + $persistent_files = !empty($extra['roundcube']['persistent-files']) + ? $extra['roundcube']['persistent-files'] + : ['config.inc.php']; foreach ($persistent_files as $file) { $path = $temp_dir . \DIRECTORY_SEPARATOR . $file; if (is_readable($path)) { diff --git a/src/PluginInstaller.php b/src/PluginInstaller.php index 6bd2c9f..e09fbe4 100644 --- a/src/PluginInstaller.php +++ b/src/PluginInstaller.php @@ -26,7 +26,9 @@ protected function confirmInstall($package_name) protected function getConfig($package_name, $config, $add) { - $cur_config = !empty($config['plugins']) ? ((array) $config['plugins']) : []; + $cur_config = !empty($config['plugins']) + ? ((array) $config['plugins']) + : []; $new_config = $cur_config; if ($add && !in_array($package_name, $new_config, true)) { @@ -36,7 +38,9 @@ protected function getConfig($package_name, $config, $add) } if ($new_config !== $cur_config) { - $config_val = count($new_config) > 0 ? "[\n\t'" . implode("',\n\t'", $new_config) . "',\n];" : '[];'; + $config_val = count($new_config) > 0 + ? "[\n\t'" . implode("',\n\t'", $new_config) . "',\n];" + : '[];'; $result = ['plugins', $config_val]; } else { $result = false; diff --git a/src/SkinInstaller.php b/src/SkinInstaller.php index 5854bf9..4d9cee5 100644 --- a/src/SkinInstaller.php +++ b/src/SkinInstaller.php @@ -26,7 +26,9 @@ protected function confirmInstall($package_name) protected function getConfig($package_name, $config, $add) { - $cur_config = !empty($config['skin']) ? $config['skin'] : null; + $cur_config = !empty($config['skin']) + ? $config['skin'] + : null; $new_config = $cur_config; if ($add && $new_config !== $package_name) { @@ -36,7 +38,9 @@ protected function getConfig($package_name, $config, $add) } if ($new_config !== $cur_config) { - $config_val = !empty($new_config) ? "'{$new_config}';" : null; + $config_val = !empty($new_config) + ? "'{$new_config}';" + : null; $result = ['skin', $config_val]; } else { $result = false; From ee7f798f83e3f3674a2a878b024cf3d91f7d480e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Vo=C5=99=C3=AD=C5=A1ek?= Date: Sun, 21 Apr 2024 01:25:34 +0200 Subject: [PATCH 4/5] Dedup initialize Roundcube environment --- src/ExtensionInstaller.php | 24 ++++++++---------------- 1 file changed, 8 insertions(+), 16 deletions(-) diff --git a/src/ExtensionInstaller.php b/src/ExtensionInstaller.php index ac3cb2e..9c5f038 100644 --- a/src/ExtensionInstaller.php +++ b/src/ExtensionInstaller.php @@ -61,16 +61,19 @@ public function getInstallPath(PackageInterface $package) . str_replace('/', \DIRECTORY_SEPARATOR, $this->getPackageName($package)); } - public function install(InstalledRepositoryInterface $repo, PackageInterface $package) + private function initializeRoundcubemailEnvironment(): void { - $this->setRoundcubemailInstallPath($repo); - // initialize Roundcube environment if (!defined('INSTALL_PATH')) { define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); } require_once INSTALL_PATH . 'program/include/iniset.php'; + } + public function install(InstalledRepositoryInterface $repo, PackageInterface $package) + { + $this->setRoundcubemailInstallPath($repo); + $this->initializeRoundcubemailEnvironment(); $this->rcubeVersionCheck($package); $postInstall = function () use ($package) { @@ -136,13 +139,7 @@ public function install(InstalledRepositoryInterface $repo, PackageInterface $pa public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target) { $this->setRoundcubemailInstallPath($repo); - - // initialize Roundcube environment - if (!defined('INSTALL_PATH')) { - define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); - } - require_once INSTALL_PATH . 'program/include/iniset.php'; - + $this->initializeRoundcubemailEnvironment(); $this->rcubeVersionCheck($target); $extra = $target->getExtra(); @@ -212,12 +209,7 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini public function uninstall(InstalledRepositoryInterface $repo, PackageInterface $package) { $this->setRoundcubemailInstallPath($repo); - - // initialize Roundcube environment - if (!defined('INSTALL_PATH')) { - define('INSTALL_PATH', $this->getRoundcubemailInstallPath() . '/'); - } - require_once INSTALL_PATH . 'program/include/iniset.php'; + $this->initializeRoundcubemailEnvironment(); $config = $this->composer->getConfig()->get('roundcube'); From d66380fb175928a88982a7e440531c5a647c6ca1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Vo=C5=99=C3=AD=C5=A1ek?= Date: Sun, 21 Apr 2024 01:40:41 +0200 Subject: [PATCH 5/5] Drop RCMAIL_VERSION < v1.2 check --- phpstan.neon.dist | 2 +- src/ExtensionInstaller.php | 14 ++------------ 2 files changed, 3 insertions(+), 13 deletions(-) diff --git a/phpstan.neon.dist b/phpstan.neon.dist index f27b06b..3812e72 100644 --- a/phpstan.neon.dist +++ b/phpstan.neon.dist @@ -16,4 +16,4 @@ parameters: - message: '~^Constant RCMAIL_VERSION not found\.$~' path: 'src/ExtensionInstaller.php' - count: 3 + count: 1 diff --git a/src/ExtensionInstaller.php b/src/ExtensionInstaller.php index 9c5f038..8cfbd61 100644 --- a/src/ExtensionInstaller.php +++ b/src/ExtensionInstaller.php @@ -108,12 +108,7 @@ public function install(InstalledRepositoryInterface $repo, PackageInterface $pa if ($sqldir = realpath($package_dir . \DIRECTORY_SEPARATOR . $extra['roundcube']['sql-dir'])) { $this->io->write("Running database initialization script for {$package_name}"); - $roundcube_version = self::versionNormalize(RCMAIL_VERSION); - if (self::versionCompare($roundcube_version, '1.2.0', '>=')) { - \rcmail_utils::db_init($sqldir); - } else { - throw new \Exception('Database initialization failed. Roundcube 1.2.0 or above required.'); - } + \rcmail_utils::db_init($sqldir); } } @@ -178,12 +173,7 @@ public function update(InstalledRepositoryInterface $repo, PackageInterface $ini if ($sqldir = realpath($package_dir . \DIRECTORY_SEPARATOR . $extra['roundcube']['sql-dir'])) { $this->io->write("Updating database schema for {$package_name}"); - $roundcube_version = self::versionNormalize(RCMAIL_VERSION); - if (self::versionCompare($roundcube_version, '1.2.0', '>=')) { - \rcmail_utils::db_update($sqldir, $package_name); - } else { - throw new \Exception('Database update failed. Roundcube 1.2.0 or above required.'); - } + \rcmail_utils::db_update($sqldir, $package_name); } }