mirror of
				https://github.com/phpbb/phpbb.git
				synced 2025-10-26 13:16:14 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			336 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			336 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  *
 | |
|  * This file is part of the phpBB Forum Software package.
 | |
|  *
 | |
|  * @copyright (c) phpBB Limited <https://www.phpbb.com>
 | |
|  * @license GNU General Public License, version 2 (GPL-2.0)
 | |
|  *
 | |
|  * For full copyright and license information, please see
 | |
|  * the docs/CREDITS.txt file.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| namespace phpbb\composer;
 | |
| 
 | |
| use Composer\IO\IOInterface;
 | |
| use phpbb\cache\driver\driver_interface;
 | |
| use phpbb\composer\exception\managed_with_clean_error_exception;
 | |
| use phpbb\composer\exception\managed_with_enable_error_exception;
 | |
| use phpbb\composer\exception\runtime_exception;
 | |
| use phpbb\config\config;
 | |
| use phpbb\extension\manager as ext_manager;
 | |
| use phpbb\filesystem\exception\filesystem_exception;
 | |
| use phpbb\filesystem\filesystem;
 | |
| 
 | |
| /**
 | |
|  * Class to safely manage extensions through composer.
 | |
|  */
 | |
| class extension_manager extends manager
 | |
| {
 | |
| 	/**
 | |
| 	 * @var ext_manager
 | |
| 	 */
 | |
| 	protected $extension_manager;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var filesystem
 | |
| 	 */
 | |
| 	protected $filesystem;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $root_path;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	private $enabled_extensions;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var bool Enables extensions when installing them?
 | |
| 	 */
 | |
| 	private $enable_on_install = false;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var bool Purges extensions data when removing them?
 | |
| 	 */
 | |
| 	private $purge_on_remove = true;
 | |
| 
 | |
| 	/**
 | |
| 	 * @param installer			$installer			Installer object
 | |
| 	 * @param driver_interface	$cache				Cache object
 | |
| 	 * @param ext_manager		$extension_manager	phpBB extension manager
 | |
| 	 * @param filesystem		$filesystem			Filesystem object
 | |
| 	 * @param string			$package_type		Composer type of managed packages
 | |
| 	 * @param string			$exception_prefix	Exception prefix to use
 | |
| 	 * @param string			$root_path			phpBB root path
 | |
| 	 * @param config|null		$config				Config object
 | |
| 	 */
 | |
| 	public function __construct(installer $installer, driver_interface $cache, ext_manager $extension_manager, filesystem $filesystem, $package_type, $exception_prefix, $root_path, config $config = null)
 | |
| 	{
 | |
| 		$this->extension_manager = $extension_manager;
 | |
| 		$this->filesystem = $filesystem;
 | |
| 		$this->root_path = $root_path;
 | |
| 
 | |
| 		if ($config)
 | |
| 		{
 | |
| 			$this->enable_on_install = (bool) $config['exts_composer_enable_on_install'];
 | |
| 			$this->purge_on_remove   = (bool) $config['exts_composer_purge_on_remove'];
 | |
| 		}
 | |
| 
 | |
| 		parent::__construct($installer, $cache, $package_type, $exception_prefix);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function pre_install(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		$installed_manually = array_intersect(array_keys($this->extension_manager->all_available()), array_keys($packages));
 | |
| 		if (count($installed_manually) !== 0)
 | |
| 		{
 | |
| 			throw new runtime_exception($this->exception_prefix, 'ALREADY_INSTALLED_MANUALLY', [implode('|', array_keys($installed_manually))]);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function post_install(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		if ($this->enable_on_install)
 | |
| 		{
 | |
| 			/** @psalm-suppress InvalidArgument */
 | |
| 			$io->writeError([['ENABLING_EXTENSIONS', [], 1]]);
 | |
| 			foreach ($packages as $package => $version)
 | |
| 			{
 | |
| 				try
 | |
| 				{
 | |
| 					$this->extension_manager->enable($package);
 | |
| 				}
 | |
| 				catch (\phpbb\exception\runtime_exception $e)
 | |
| 				{
 | |
| 					/** @psalm-suppress InvalidArgument */
 | |
| 					$io->writeError([[$e->getMessage(), $e->get_parameters(), 4]]);
 | |
| 				}
 | |
| 				catch (\Exception $e)
 | |
| 				{
 | |
| 					/** @psalm-suppress InvalidArgument */
 | |
| 					$io->writeError([[$e->getMessage(), [], 4]]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	protected function pre_update(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		/** @psalm-suppress InvalidArgument */
 | |
| 		$io->writeError([['DISABLING_EXTENSIONS', [], 1]]);
 | |
| 		$this->enabled_extensions = [];
 | |
| 		foreach ($packages as $package => $version)
 | |
| 		{
 | |
| 			try
 | |
| 			{
 | |
| 				if ($this->extension_manager->is_enabled($package))
 | |
| 				{
 | |
| 					$this->enabled_extensions[] = $package;
 | |
| 					$this->extension_manager->disable($package);
 | |
| 				}
 | |
| 			}
 | |
| 			catch (\phpbb\exception\runtime_exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), $e->get_parameters(), 4]]);
 | |
| 			}
 | |
| 			catch (\Exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), [], 4]]);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	protected function post_update(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		/** @psalm-suppress InvalidArgument */
 | |
| 		$io->writeError([['ENABLING_EXTENSIONS', [], 1]]);
 | |
| 		foreach ($this->enabled_extensions as $package)
 | |
| 		{
 | |
| 			try
 | |
| 			{
 | |
| 				$this->extension_manager->enable($package);
 | |
| 			}
 | |
| 			catch (\phpbb\exception\runtime_exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), $e->get_parameters(), 4]]);
 | |
| 			}
 | |
| 			catch (\Exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), [], 4]]);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function remove(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		$packages = $this->normalize_version($packages);
 | |
| 
 | |
| 		$not_installed = array_diff(array_keys($packages), array_keys($this->extension_manager->all_available()));
 | |
| 		if (count($not_installed) !== 0)
 | |
| 		{
 | |
| 			throw new runtime_exception($this->exception_prefix, 'NOT_INSTALLED', [implode('|', array_keys($not_installed))]);
 | |
| 		}
 | |
| 
 | |
| 		parent::remove($packages, $io);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function pre_remove(array $packages, IOInterface $io = null)
 | |
| 	{
 | |
| 		if ($this->purge_on_remove)
 | |
| 		{
 | |
| 			/** @psalm-suppress InvalidArgument */
 | |
| 			$io->writeError([['DISABLING_EXTENSIONS', [], 1]]);
 | |
| 		}
 | |
| 
 | |
| 		foreach ($packages as $package => $version)
 | |
| 		{
 | |
| 			try
 | |
| 			{
 | |
| 				if ($this->extension_manager->is_enabled($package))
 | |
| 				{
 | |
| 					if ($this->purge_on_remove)
 | |
| 					{
 | |
| 						$this->extension_manager->purge($package);
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						$this->extension_manager->disable($package);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			catch (\phpbb\exception\runtime_exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), $e->get_parameters(), 4]]);
 | |
| 			}
 | |
| 			catch (\Exception $e)
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([[$e->getMessage(), [], 4]]);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function start_managing($package, $io)
 | |
| 	{
 | |
| 		if (!$this->extension_manager->is_available($package))
 | |
| 		{
 | |
| 			throw new runtime_exception($this->exception_prefix, 'NOT_INSTALLED', [$package]);
 | |
| 		}
 | |
| 
 | |
| 		if ($this->is_managed($package))
 | |
| 		{
 | |
| 			throw new runtime_exception($this->exception_prefix, 'ALREADY_MANAGED', [$package]);
 | |
| 		}
 | |
| 
 | |
| 		$enabled = false;
 | |
| 		if ($this->extension_manager->is_enabled($package))
 | |
| 		{
 | |
| 			$enabled = true;
 | |
| 			/** @psalm-suppress InvalidArgument */
 | |
| 			$io->writeError([['DISABLING_EXTENSIONS', [], 1]]);
 | |
| 			$this->extension_manager->disable($package);
 | |
| 		}
 | |
| 
 | |
| 		$ext_path = $this->extension_manager->get_extension_path($package, true);
 | |
| 		$backup_path = rtrim($ext_path, '/') . '__backup__';
 | |
| 
 | |
| 		try
 | |
| 		{
 | |
| 			$this->filesystem->rename($ext_path, $backup_path);
 | |
| 		}
 | |
| 		catch (filesystem_exception $e)
 | |
| 		{
 | |
| 			throw new runtime_exception($this->exception_prefix, 'CANNOT_MANAGE_FILESYSTEM_ERROR', [$package], $e);
 | |
| 		}
 | |
| 
 | |
| 		try
 | |
| 		{
 | |
| 			$this->install((array) $package, $io);
 | |
| 			$this->filesystem->remove($backup_path);
 | |
| 		}
 | |
| 		catch (runtime_exception $e)
 | |
| 		{
 | |
| 			$this->filesystem->rename($backup_path, $ext_path);
 | |
| 			throw new runtime_exception($this->exception_prefix, 'CANNOT_MANAGE_INSTALL_ERROR', [$package], $e);
 | |
| 		}
 | |
| 		catch (filesystem_exception $e)
 | |
| 		{
 | |
| 			throw new managed_with_clean_error_exception($this->exception_prefix, 'MANAGED_WITH_CLEAN_ERROR', [$package, $backup_path], $e);
 | |
| 		}
 | |
| 
 | |
| 		if ($enabled)
 | |
| 		{
 | |
| 			try
 | |
| 			{
 | |
| 				/** @psalm-suppress InvalidArgument */
 | |
| 				$io->writeError([['ENABLING_EXTENSIONS', [], 1]]);
 | |
| 				$this->extension_manager->enable($package);
 | |
| 			}
 | |
| 			catch (\Exception $e)
 | |
| 			{
 | |
| 				throw new managed_with_enable_error_exception($this->exception_prefix, 'MANAGED_WITH_ENABLE_ERROR', [$package], $e);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * {@inheritdoc}
 | |
| 	 */
 | |
| 	public function check_requirements()
 | |
| 	{
 | |
| 		return parent::check_requirements() && $this->filesystem->is_writable($this->root_path . 'ext/');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Enable the extensions when installing
 | |
| 	 *
 | |
| 	 * Warning: Only the explicitly required extensions will be enabled
 | |
| 	 *
 | |
| 	 * @param bool $enable
 | |
| 	 */
 | |
| 	public function set_enable_on_install($enable)
 | |
| 	{
 | |
| 		$this->enable_on_install = $enable;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Purge the extension when disabling it
 | |
| 	 *
 | |
| 	 * @param bool $purge
 | |
| 	 */
 | |
| 	public function set_purge_on_remove($purge)
 | |
| 	{
 | |
| 		$this->purge_on_remove = $purge;
 | |
| 	}
 | |
| }
 |