diff options
Diffstat (limited to 'phpBB/phpbb/console')
| -rw-r--r-- | phpBB/phpbb/console/application.php | 71 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/cron/run.php | 7 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/db/list_command.php | 73 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/db/migrate.php | 45 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/db/migration_command.php | 56 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/db/revert.php | 83 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/reparser/list_all.php | 69 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/reparser/reparse.php | 305 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/thumbnail/delete.php | 178 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/thumbnail/generate.php | 204 | ||||
| -rw-r--r-- | phpBB/phpbb/console/command/thumbnail/recreate.php | 72 | ||||
| -rw-r--r-- | phpBB/phpbb/console/exception_subscriber.php | 74 | 
12 files changed, 1177 insertions, 60 deletions
| diff --git a/phpBB/phpbb/console/application.php b/phpBB/phpbb/console/application.php index bc4897af18..dc9b8016b2 100644 --- a/phpBB/phpbb/console/application.php +++ b/phpBB/phpbb/console/application.php @@ -13,6 +13,7 @@  namespace phpbb\console; +use Symfony\Component\Console\Input\InputDefinition;  use Symfony\Component\Console\Shell;  use Symfony\Component\Console\Input\InputInterface;  use Symfony\Component\Console\Input\InputOption; @@ -26,18 +27,18 @@ class application extends \Symfony\Component\Console\Application  	protected $in_shell = false;  	/** -	* @var \phpbb\user User object +	* @var \phpbb\language\language User object  	*/ -	protected $user; +	protected $language;  	/** -	* @param string			$name		The name of the application -	* @param string			$version	The version of the application -	* @param \phpbb\user	$user		The user which runs the application (used for translation) +	* @param string						$name		The name of the application +	* @param string						$version	The version of the application +	* @param \phpbb\language\language	$language	The user which runs the application (used for translation)  	*/ -	public function __construct($name, $version, \phpbb\user $user) +	public function __construct($name, $version, \phpbb\language\language $language)  	{ -		$this->user = $user; +		$this->language = $language;  		parent::__construct($name, $version);  	} @@ -49,12 +50,7 @@ class application extends \Symfony\Component\Console\Application  	{  		$input_definition = parent::getDefaultInputDefinition(); -		$input_definition->addOption(new InputOption( -			'safe-mode', -			null, -			InputOption::VALUE_NONE, -			$this->user->lang('CLI_DESCRIPTION_OPTION_SAFE_MODE') -		)); +		$this->register_global_options($input_definition);  		return $input_definition;  	} @@ -76,12 +72,20 @@ class application extends \Symfony\Component\Console\Application  			return parent::getHelp();  		} -		$this->getDefinition()->addOption(new InputOption( -			'--shell', -			'-s', -			InputOption::VALUE_NONE, -			$this->user->lang('CLI_DESCRIPTION_OPTION_SHELL') -		)); +		try +		{ +			$definition = $this->getDefinition(); +			$definition->addOption(new InputOption( +				'--shell', +				'-s', +				InputOption::VALUE_NONE, +				$this->language->lang('CLI_DESCRIPTION_OPTION_SHELL') +			)); +		} +		catch (\LogicException $e) +		{ +			// Do nothing +		}  		return parent::getHelp();  	} @@ -117,4 +121,33 @@ class application extends \Symfony\Component\Console\Application  		return parent::doRun($input, $output);  	} + +	/** +	 * Register global options +	 * +	 * @param InputDefinition $definition An InputDefinition instance +	 */ +	protected function register_global_options(InputDefinition $definition) +	{ +		try +		{ +			$definition->addOption(new InputOption( +				'safe-mode', +				null, +				InputOption::VALUE_NONE, +				$this->language->lang('CLI_DESCRIPTION_OPTION_SAFE_MODE') +			)); + +			$definition->addOption(new InputOption( +				'env', +				'e', +				InputOption::VALUE_REQUIRED, +				$this->language->lang('CLI_DESCRIPTION_OPTION_ENV') +			)); +		} +		catch (\LogicException $e) +		{ +			// Do nothing +		} +	}  } diff --git a/phpBB/phpbb/console/command/cron/run.php b/phpBB/phpbb/console/command/cron/run.php index a9648fcd41..dea6493007 100644 --- a/phpBB/phpbb/console/command/cron/run.php +++ b/phpBB/phpbb/console/command/cron/run.php @@ -13,6 +13,7 @@  namespace phpbb\console\command\cron; +use phpbb\exception\runtime_exception;  use Symfony\Component\Console\Input\InputInterface;  use Symfony\Component\Console\Input\InputArgument;  use Symfony\Component\Console\Output\OutputInterface; @@ -93,8 +94,7 @@ class run extends \phpbb\console\command\command  		}  		else  		{ -			$output->writeln('<error>' . $this->user->lang('CRON_LOCK_ERROR') . '</error>'); -			return 1; +			throw new runtime_exception('CRON_LOCK_ERROR', array(), null, 1);  		}  	} @@ -165,8 +165,7 @@ class run extends \phpbb\console\command\command  		}  		else  		{ -			$output->writeln('<error>' . $this->user->lang('CRON_NO_SUCH_TASK', $task_name) . '</error>'); -			return 2; +			throw new runtime_exception('CRON_NO_SUCH_TASK', array( $task_name), null, 2);  		}  	}  } diff --git a/phpBB/phpbb/console/command/db/list_command.php b/phpBB/phpbb/console/command/db/list_command.php new file mode 100644 index 0000000000..708107b592 --- /dev/null +++ b/phpBB/phpbb/console/command/db/list_command.php @@ -0,0 +1,73 @@ +<?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\console\command\db; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputOption; +use Symfony\Component\Console\Output\OutputInterface; + +class list_command extends \phpbb\console\command\db\migration_command +{ +	protected function configure() +	{ +		$this +			->setName('db:list') +			->setDescription($this->user->lang('CLI_DESCRIPTION_DB_LIST')) +			->addOption( +				'available', +				'u', +				InputOption::VALUE_NONE, +				$this->user->lang('CLI_MIGRATIONS_ONLY_AVAILABLE') +			) +		; +	} + +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$show_installed = !$input->getOption('available'); +		$installed = $available = array(); + +		foreach ($this->load_migrations() as $name) +		{ +			if ($this->migrator->migration_state($name) !== false) +			{ +				$installed[] = $name; +			} +			else +			{ +				$available[] = $name; +			} +		} + +		if ($show_installed) +		{ +			$output->writeln('<info>' . $this->user->lang('CLI_MIGRATIONS_INSTALLED') . $this->user->lang('COLON') . '</info>'); +			$output->writeln($installed); + +			if (empty($installed)) +			{ +				$output->writeln($this->user->lang('CLI_MIGRATIONS_EMPTY')); +			} + +			$output->writeln(''); +		} + +		$output->writeln('<info>' . $this->user->lang('CLI_MIGRATIONS_AVAILABLE') . $this->user->lang('COLON') . '</info>'); +		$output->writeln($available); + +		if (empty($available)) +		{ +			$output->writeln($this->user->lang('CLI_MIGRATIONS_EMPTY')); +		} +	} +} diff --git a/phpBB/phpbb/console/command/db/migrate.php b/phpBB/phpbb/console/command/db/migrate.php index 87c2a057d1..43029b7458 100644 --- a/phpBB/phpbb/console/command/db/migrate.php +++ b/phpBB/phpbb/console/command/db/migrate.php @@ -15,35 +15,23 @@ namespace phpbb\console\command\db;  use Symfony\Component\Console\Input\InputInterface;  use Symfony\Component\Console\Output\OutputInterface; -class migrate extends \phpbb\console\command\command +class migrate extends \phpbb\console\command\db\migration_command  { -	/** @var \phpbb\db\migrator */ -	protected $migrator; - -	/** @var \phpbb\extension\manager */ -	protected $extension_manager; - -	/** @var \phpbb\config\config */ -	protected $config; - -	/** @var \phpbb\cache\service */ -	protected $cache; -  	/** @var \phpbb\log\log */  	protected $log;  	/** @var string phpBB root path */  	protected $phpbb_root_path; -	function __construct(\phpbb\user $user, \phpbb\db\migrator $migrator, \phpbb\extension\manager $extension_manager, \phpbb\config\config $config, \phpbb\cache\service $cache, \phpbb\log\log $log, $phpbb_root_path) +	/** @var  \phpbb\filesystem\filesystem_interface */ +	protected $filesystem; + +	function __construct(\phpbb\user $user, \phpbb\db\migrator $migrator, \phpbb\extension\manager $extension_manager, \phpbb\config\config $config, \phpbb\cache\service $cache, \phpbb\log\log $log, \phpbb\filesystem\filesystem_interface $filesystem, $phpbb_root_path)  	{ -		$this->migrator = $migrator; -		$this->extension_manager = $extension_manager; -		$this->config = $config; -		$this->cache = $cache;  		$this->log = $log; +		$this->filesystem = $filesystem;  		$this->phpbb_root_path = $phpbb_root_path; -		parent::__construct($user); +		parent::__construct($user, $migrator, $extension_manager, $config, $cache);  		$this->user->add_lang(array('common', 'install', 'migrator'));  	} @@ -57,7 +45,7 @@ class migrate extends \phpbb\console\command\command  	protected function execute(InputInterface $input, OutputInterface $output)  	{ -		$this->migrator->set_output_handler(new \phpbb\db\log_wrapper_migrator_output_handler($this->user, new console_migrator_output_handler($this->user, $output), $this->phpbb_root_path . 'store/migrations_' . time() . '.log')); +		$this->migrator->set_output_handler(new \phpbb\db\log_wrapper_migrator_output_handler($this->user, new console_migrator_output_handler($this->user, $output), $this->phpbb_root_path . 'store/migrations_' . time() . '.log', $this->filesystem));  		$this->migrator->create_migrations_table(); @@ -87,21 +75,4 @@ class migrate extends \phpbb\console\command\command  		$this->finalise_update();  		$output->writeln($this->user->lang['DATABASE_UPDATE_COMPLETE']);  	} - -	protected function load_migrations() -	{ -		$migrations = $this->extension_manager -			->get_finder() -			->core_path('phpbb/db/migration/data/') -			->extension_directory('/migrations') -			->get_classes(); - -		$this->migrator->set_migrations($migrations); -	} - -	protected function finalise_update() -	{ -		$this->cache->purge(); -		$this->config->increment('assets_version', 1); -	}  } diff --git a/phpBB/phpbb/console/command/db/migration_command.php b/phpBB/phpbb/console/command/db/migration_command.php new file mode 100644 index 0000000000..d44ef8c5cb --- /dev/null +++ b/phpBB/phpbb/console/command/db/migration_command.php @@ -0,0 +1,56 @@ +<?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\console\command\db; + +abstract class migration_command extends \phpbb\console\command\command +{ +	/** @var \phpbb\db\migrator */ +	protected $migrator; + +	/** @var \phpbb\extension\manager */ +	protected $extension_manager; + +	/** @var \phpbb\config\config */ +	protected $config; + +	/** @var \phpbb\cache\service */ +	protected $cache; + +	function __construct(\phpbb\user $user, \phpbb\db\migrator $migrator, \phpbb\extension\manager $extension_manager, \phpbb\config\config $config, \phpbb\cache\service $cache) +	{ +		$this->migrator = $migrator; +		$this->extension_manager = $extension_manager; +		$this->config = $config; +		$this->cache = $cache; +		parent::__construct($user); +	} + +	protected function load_migrations() +	{ +		$migrations = $this->extension_manager +			->get_finder() +			->core_path('phpbb/db/migration/data/') +			->extension_directory('/migrations') +			->get_classes(); + +		$this->migrator->set_migrations($migrations); + +		return $migrations; +	} + +	protected function finalise_update() +	{ +		$this->cache->purge(); +		$this->config->increment('assets_version', 1); +	} +} diff --git a/phpBB/phpbb/console/command/db/revert.php b/phpBB/phpbb/console/command/db/revert.php new file mode 100644 index 0000000000..838640968e --- /dev/null +++ b/phpBB/phpbb/console/command/db/revert.php @@ -0,0 +1,83 @@ +<?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\console\command\db; + +use Symfony\Component\Console\Input\InputArgument; +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Output\OutputInterface; + +class revert extends \phpbb\console\command\db\migration_command +{ +	/** @var string phpBB root path */ +	protected $phpbb_root_path; + +	/** @var  \phpbb\filesystem\filesystem_interface */ +	protected $filesystem; + +	function __construct(\phpbb\user $user, \phpbb\db\migrator $migrator, \phpbb\extension\manager $extension_manager, \phpbb\config\config $config, \phpbb\cache\service $cache, \phpbb\filesystem\filesystem_interface $filesystem, $phpbb_root_path) +	{ +		$this->filesystem = $filesystem; +		$this->phpbb_root_path = $phpbb_root_path; +		parent::__construct($user, $migrator, $extension_manager, $config, $cache); +		$this->user->add_lang(array('common', 'migrator')); +	} + +	protected function configure() +	{ +		$this +			->setName('db:revert') +			->setDescription($this->user->lang('CLI_DESCRIPTION_DB_REVERT')) +			->addArgument( +				'name', +				InputArgument::REQUIRED, +				$this->user->lang('CLI_MIGRATION_NAME') +			) +		; +	} + +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$name = str_replace('/', '\\', $input->getArgument('name')); + +		$this->migrator->set_output_handler(new \phpbb\db\log_wrapper_migrator_output_handler($this->user, new console_migrator_output_handler($this->user, $output), $this->phpbb_root_path . 'store/migrations_' . time() . '.log', $this->filesystem)); + +		$this->cache->purge(); + +		if (!in_array($name, $this->load_migrations())) +		{ +			$output->writeln('<error>' . $this->user->lang('MIGRATION_NOT_VALID', $name) . '</error>'); +			return 1; +		} +		else if ($this->migrator->migration_state($name) === false) +		{ +			$output->writeln('<error>' . $this->user->lang('MIGRATION_NOT_INSTALLED', $name) . '</error>'); +			return 1; +		} + +		try +		{ +			while ($this->migrator->migration_state($name) !== false) +			{ +				$this->migrator->revert($name); +			} +		} +		catch (\phpbb\db\migration\exception $e) +		{ +			$output->writeln('<error>' . $e->getLocalisedMessage($this->user) . '</error>'); +			$this->finalise_update(); +			return 1; +		} + +		$this->finalise_update(); +	} +} diff --git a/phpBB/phpbb/console/command/reparser/list_all.php b/phpBB/phpbb/console/command/reparser/list_all.php new file mode 100644 index 0000000000..e42c3ac782 --- /dev/null +++ b/phpBB/phpbb/console/command/reparser/list_all.php @@ -0,0 +1,69 @@ +<?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\console\command\reparser; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Output\OutputInterface; + +class list_all extends \phpbb\console\command\command +{ +	/** +	* @var string[] Names of the reparser services +	*/ +	protected $reparser_names; + +	/** +	* Constructor +	* +	* @param \phpbb\user $user +	* @param \phpbb\di\service_collection $reparsers +	*/ +	public function __construct(\phpbb\user $user, \phpbb\di\service_collection $reparsers) +	{ +		parent::__construct($user); +		$this->reparser_names = array(); +		foreach ($reparsers as $name => $reparser) +		{ +			// Store the names without the "text_reparser." prefix +			$this->reparser_names[] = preg_replace('(^text_reparser\\.)', '', $name); +		} +	} + +	/** +	* Sets the command name and description +	* +	* @return null +	*/ +	protected function configure() +	{ +		$this +			->setName('reparser:list') +			->setDescription($this->user->lang('CLI_DESCRIPTION_REPARSER_LIST')) +		; +	} + +	/** +	* Executes the command reparser:reparse +	* +	* @param InputInterface $input +	* @param OutputInterface $output +	* @return integer +	*/ +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$output->writeln('<info>' . implode(', ', $this->reparser_names) . '</info>'); + +		return 0; +	} +} diff --git a/phpBB/phpbb/console/command/reparser/reparse.php b/phpBB/phpbb/console/command/reparser/reparse.php new file mode 100644 index 0000000000..63124b4b8c --- /dev/null +++ b/phpBB/phpbb/console/command/reparser/reparse.php @@ -0,0 +1,305 @@ +<?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\console\command\reparser; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputArgument; +use Symfony\Component\Console\Input\InputOption; +use Symfony\Component\Console\Output\OutputInterface; +use Symfony\Component\Console\Style\SymfonyStyle; + +class reparse extends \phpbb\console\command\command +{ +	/** +	* @var \phpbb\config\db_text +	*/ +	protected $config_text; + +	/** +	* @var InputInterface +	*/ +	protected $input; + +	/** +	* @var SymfonyStyle +	*/ +	protected $io; + +	/** +	* @var OutputInterface +	*/ +	protected $output; + +	/** +	* @var \phpbb\di\service_collection +	*/ +	protected $reparsers; + +	/** +	* @var array Reparser names as keys, and their last $current ID as values +	*/ +	protected $resume_data; + +	/** +	* Constructor +	* +	* @param \phpbb\user $user +	* @param \phpbb\di\service_collection $reparsers +	* @param \phpbb\config\db_text $config_text +	*/ +	public function __construct(\phpbb\user $user, \phpbb\di\service_collection $reparsers, \phpbb\config\db_text $config_text) +	{ +		require_once __DIR__ . '/../../../../includes/functions_content.php'; + +		$this->config_text = $config_text; +		$this->reparsers = $reparsers; +		parent::__construct($user); +	} + +	/** +	* Sets the command name and description +	* +	* @return null +	*/ +	protected function configure() +	{ +		$this +			->setName('reparser:reparse') +			->setDescription($this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE')) +			->addArgument('reparser-name', InputArgument::OPTIONAL, $this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_ARG_1')) +			->addOption( +				'dry-run', +				null, +				InputOption::VALUE_NONE, +				$this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_OPT_DRY_RUN') +			) +			->addOption( +				'resume', +				null, +				InputOption::VALUE_NONE, +				$this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_OPT_RESUME') +			) +			->addOption( +				'range-min', +				null, +				InputOption::VALUE_REQUIRED, +				$this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_OPT_RANGE_MIN'), +				1 +			) +			->addOption( +				'range-max', +				null, +				InputOption::VALUE_REQUIRED, +				$this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_OPT_RANGE_MAX') +			) +			->addOption( +				'range-size', +				null, +				InputOption::VALUE_REQUIRED, +				$this->user->lang('CLI_DESCRIPTION_REPARSER_REPARSE_OPT_RANGE_SIZE'), +				100 +			); +		; +	} + +	/** +	* Create a styled progress bar +	* +	* @param  integer $max Max value for the progress bar +	* @return \Symfony\Component\Console\Helper\ProgressBar +	*/ +	protected function create_progress_bar($max) +	{ +		$progress = $this->io->createProgressBar($max); +		if ($this->output->getVerbosity() === OutputInterface::VERBOSITY_VERBOSE) +		{ +			$progress->setFormat('<info>[%percent:3s%%]</info> %message%'); +			$progress->setOverwrite(false); +		} +		else if ($this->output->getVerbosity() >= OutputInterface::VERBOSITY_VERY_VERBOSE) +		{ +			$progress->setFormat('<info>[%current:s%/%max:s%]</info><comment>[%elapsed%/%estimated%][%memory%]</comment> %message%'); +			$progress->setOverwrite(false); +		} +		else +		{ +			$this->io->newLine(2); +			$progress->setFormat( +				"    %current:s%/%max:s% %bar%  %percent:3s%%\n" . +				"        %message% %elapsed:6s%/%estimated:-6s% %memory:6s%\n"); +			$progress->setBarWidth(60); +		} + +		if (!defined('PHP_WINDOWS_VERSION_BUILD')) +		{ +			$progress->setEmptyBarCharacter('░'); // light shade character \u2591 +			$progress->setProgressCharacter(''); +			$progress->setBarCharacter('▓'); // dark shade character \u2593 +		} + +		return $progress; +	} + +	/** +	* Executes the command reparser:reparse +	* +	* @param InputInterface $input +	* @param OutputInterface $output +	* @return integer +	*/ +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$this->input = $input; +		$this->output = $output; +		$this->io = new SymfonyStyle($input, $output); +		$this->load_resume_data(); + +		$name = $input->getArgument('reparser-name'); +		if (isset($name)) +		{ +			// Allow "post_text" to be an alias for "text_reparser.post_text" +			if (!isset($this->reparsers[$name])) +			{ +				$name = 'text_reparser.' . $name; +			} +			$this->reparse($name); +		} +		else +		{ +			foreach ($this->reparsers as $name => $service) +			{ +				$this->reparse($name); +			} +		} + +		$this->io->success($this->user->lang('CLI_REPARSER_REPARSE_SUCCESS')); + +		return 0; +	} + +	/** +	* Get an option value, adjusted for given reparser +	* +	* Will use the last saved value if --resume is set and the option was not specified +	* on the command line +	* +	* @param  string  $reparser_name Reparser name +	* @param  string  $option_name   Option name +	* @return integer +	*/ +	protected function get_option($reparser_name, $option_name) +	{ +		// Return the option from the resume_data if applicable +		if ($this->input->getOption('resume') && isset($this->resume_data[$reparser_name][$option_name]) && !$this->input->hasParameterOption('--' . $option_name)) +		{ +			return $this->resume_data[$reparser_name][$option_name]; +		} + +		$value = $this->input->getOption($option_name); + +		// range-max has no default value, it must be computed for each reparser +		if ($option_name === 'range-max' && $value === null) +		{ +			$value = $this->reparsers[$reparser_name]->get_max_id(); +		} + +		return $value; +	} + +	/** +	* Load the resume data from the database +	*/ +	protected function load_resume_data() +	{ +		$resume_data = $this->config_text->get('reparser_resume'); +		$this->resume_data = (empty($resume_data)) ? array() : unserialize($resume_data); +	} + +	/** +	* Reparse all text handled by given reparser within given range +	* +	* @param string $name Reparser name +	*/ +	protected function reparse($name) +	{ +		$reparser = $this->reparsers[$name]; +		if ($this->input->getOption('dry-run')) +		{ +			$reparser->disable_save(); +		} +		else +		{ +			$reparser->enable_save(); +		} + +		// Start at range-max if specified or at the highest ID otherwise +		$max  = $this->get_option($name, 'range-max'); +		$min  = $this->get_option($name, 'range-min'); +		$size = $this->get_option($name, 'range-size'); + +		if ($max < $min) +		{ +			return; +		} + +		$this->io->section($this->user->lang('CLI_REPARSER_REPARSE_REPARSING', preg_replace('(^text_reparser\\.)', '', $name), $min, $max)); + +		$progress = $this->create_progress_bar($max); +		$progress->setMessage($this->user->lang('CLI_REPARSER_REPARSE_REPARSING_START', preg_replace('(^text_reparser\\.)', '', $name))); +		$progress->start(); + +		// Start from $max and decrement $current by $size until we reach $min +		$current = $max; +		while ($current >= $min) +		{ +			$start = max($min, $current + 1 - $size); +			$end   = max($min, $current); + +			$progress->setMessage($this->user->lang('CLI_REPARSER_REPARSE_REPARSING', preg_replace('(^text_reparser\\.)', '', $name), $start, $end)); +			$reparser->reparse_range($start, $end); + +			$current = $start - 1; +			$progress->setProgress($max + 1 - $start); + +			$this->update_resume_data($name, $current); +		} +		$progress->finish(); + +		$this->io->newLine(2); +	} + +	/** +	* Save the resume data to the database +	*/ +	protected function save_resume_data() +	{ +		$this->config_text->set('reparser_resume', serialize($this->resume_data)); +	} + +	/** +	* Save the resume data to the database +	* +	* @param string $name    Reparser name +	* @param string $current Current ID +	*/ +	protected function update_resume_data($name, $current) +	{ +		$this->resume_data[$name] = array( +			'range-min'  => $this->get_option($name, 'range-min'), +			'range-max'  => $current, +			'range-size' => $this->get_option($name, 'range-size'), +		); +		$this->save_resume_data(); +	} +} diff --git a/phpBB/phpbb/console/command/thumbnail/delete.php b/phpBB/phpbb/console/command/thumbnail/delete.php new file mode 100644 index 0000000000..e8e4cf568e --- /dev/null +++ b/phpBB/phpbb/console/command/thumbnail/delete.php @@ -0,0 +1,178 @@ +<?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\console\command\thumbnail; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Output\OutputInterface; +use Symfony\Component\Console\Style\SymfonyStyle; + +class delete extends \phpbb\console\command\command +{ +	/** +	* @var \phpbb\db\driver\driver_interface +	*/ +	protected $db; + +	/** +	* phpBB root path +	* @var string +	*/ +	protected $phpbb_root_path; + +	/** +	* Constructor +	* +	* @param \phpbb\user $user The user object (used to get language information) +	* @param \phpbb\db\driver\driver_interface $db Database connection +	* @param string $phpbb_root_path Root path +	*/ +	public function __construct(\phpbb\user $user, \phpbb\db\driver\driver_interface $db, $phpbb_root_path) +	{ +		$this->db = $db; +		$this->phpbb_root_path = $phpbb_root_path; + +		parent::__construct($user); +	} + +	/** +	* Sets the command name and description +	* +	* @return null +	*/ +	protected function configure() +	{ +		$this +			->setName('thumbnail:delete') +			->setDescription($this->user->lang('CLI_DESCRIPTION_THUMBNAIL_DELETE')) +		; +	} + +	/** +	* Executes the command thumbnail:delete. +	* +	* Deletes all existing thumbnails and updates the database accordingly. +	* +	* @param InputInterface $input The input stream used to get the argument and verbose option. +	* @param OutputInterface $output The output stream, used for printing verbose-mode and error information. +	* +	* @return int 0 if all is ok, 1 if a thumbnail couldn't be deleted. +	*/ +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$io = new SymfonyStyle($input, $output); + +		$io->section($this->user->lang('CLI_THUMBNAIL_DELETING')); + +		$sql = 'SELECT COUNT(*) AS nb_missing_thumbnails +			FROM ' . ATTACHMENTS_TABLE . ' +			WHERE thumbnail = 1'; +		$result = $this->db->sql_query($sql); +		$nb_missing_thumbnails = (int) $this->db->sql_fetchfield('nb_missing_thumbnails'); +		$this->db->sql_freeresult($result); + +		if ($nb_missing_thumbnails === 0) +		{ +			$io->warning($this->user->lang('CLI_THUMBNAIL_NOTHING_TO_DELETE')); +			return 0; +		} + +		$sql = 'SELECT attach_id, physical_filename, extension, real_filename, mimetype +			FROM ' . ATTACHMENTS_TABLE . ' +			WHERE thumbnail = 1'; +		$result = $this->db->sql_query($sql); + +		$progress = $io->createProgressBar($nb_missing_thumbnails); +		if ($output->getVerbosity() === OutputInterface::VERBOSITY_VERBOSE) +		{ +			$progress->setFormat('<info>[%percent:3s%%]</info> %message%'); +			$progress->setOverwrite(false); +		} +		else if ($output->getVerbosity() >= OutputInterface::VERBOSITY_VERY_VERBOSE) +		{ +			$progress->setFormat('<info>[%current:s%/%max:s%]</info><comment>[%elapsed%/%estimated%][%memory%]</comment> %message%'); +			$progress->setOverwrite(false); +		} +		else +		{ +			$io->newLine(2); +			$progress->setFormat( +				"    %current:s%/%max:s% %bar%  %percent:3s%%\n" . +				"                         %elapsed:6s%/%estimated:-6s% %memory:6s%\n"); +			$progress->setBarWidth(60); +		} + +		if (!defined('PHP_WINDOWS_VERSION_BUILD')) +		{ +			$progress->setEmptyBarCharacter('░'); // light shade character \u2591 +			$progress->setProgressCharacter(''); +			$progress->setBarCharacter('▓'); // dark shade character \u2593 +		} + +		$progress->setMessage($this->user->lang('CLI_THUMBNAIL_DELETING')); + +		$progress->start(); + +		$thumbnail_deleted = array(); +		$return = 0; +		while ($row = $this->db->sql_fetchrow($result)) +		{ +			$thumbnail_path = $this->phpbb_root_path . 'files/thumb_' . $row['physical_filename']; + +			if (@unlink($thumbnail_path)) +			{ +				$thumbnail_deleted[] = $row['attach_id']; + +				if (sizeof($thumbnail_deleted) === 250) +				{ +					$this->commit_changes($thumbnail_deleted); +					$thumbnail_deleted = array(); +				} + +				$progress->setMessage($this->user->lang('CLI_THUMBNAIL_DELETED', $row['real_filename'], $row['physical_filename'])); +			} +			else +			{ +				$return = 1; +				$progress->setMessage('<error>' . $this->user->lang('CLI_THUMBNAIL_SKIPPED', $row['real_filename'], $row['physical_filename']) . '</error>'); +			} + +			$progress->advance(); +		} +		$this->db->sql_freeresult($result); + +		if (!empty($thumbnail_deleted)) +		{ +			$this->commit_changes($thumbnail_deleted); +		} + +		$progress->finish(); + +		$io->newLine(2); +		$io->success($this->user->lang('CLI_THUMBNAIL_DELETING_DONE')); + +		return $return; +	} + +	/** +	* Commits the changes to the database +	* +	* @param array $thumbnail_deleted +	*/ +	protected function commit_changes(array $thumbnail_deleted) +	{ +		$sql = 'UPDATE ' . ATTACHMENTS_TABLE . ' +				SET thumbnail = 0 +				WHERE ' . $this->db->sql_in_set('attach_id', $thumbnail_deleted); +		$this->db->sql_query($sql); +	} +} diff --git a/phpBB/phpbb/console/command/thumbnail/generate.php b/phpBB/phpbb/console/command/thumbnail/generate.php new file mode 100644 index 0000000000..e677db3a97 --- /dev/null +++ b/phpBB/phpbb/console/command/thumbnail/generate.php @@ -0,0 +1,204 @@ +<?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\console\command\thumbnail; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Output\OutputInterface; +use Symfony\Component\Console\Style\SymfonyStyle; + +class generate extends \phpbb\console\command\command +{ +	/** +	* @var \phpbb\db\driver\driver_interface +	*/ +	protected $db; + +	/** +	* @var \phpbb\cache\service +	*/ +	protected $cache; + +	/** +	* phpBB root path +	* @var string +	*/ +	protected $phpbb_root_path; + +	/** +	* PHP extension. +	* +	* @var string +	*/ +	protected $php_ext; + +	/** +	* Constructor +	* +	* @param \phpbb\user $user The user object (used to get language information) +	* @param \phpbb\db\driver\driver_interface $db Database connection +	* @param \phpbb\cache\service $cache The cache service +	* @param string $phpbb_root_path Root path +	* @param string $php_ext PHP extension +	*/ +	public function __construct(\phpbb\user $user, \phpbb\db\driver\driver_interface $db, \phpbb\cache\service $cache, $phpbb_root_path, $php_ext) +	{ +		$this->db = $db; +		$this->cache = $cache; +		$this->phpbb_root_path = $phpbb_root_path; +		$this->php_ext = $php_ext; + +		parent::__construct($user); +	} + +	/** +	* Sets the command name and description +	* +	* @return null +	*/ +	protected function configure() +	{ +		$this +			->setName('thumbnail:generate') +			->setDescription($this->user->lang('CLI_DESCRIPTION_THUMBNAIL_GENERATE')) +		; +	} + +	/** +	* Executes the command thumbnail:generate. +	* +	* Generate a thumbnail for all attachments which need one and don't have it yet. +	* +	* @param InputInterface $input The input stream used to get the argument and verboe option. +	* @param OutputInterface $output The output stream, used for printing verbose-mode and error information. +	* +	* @return int 0. +	*/ +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$io = new SymfonyStyle($input, $output); + +		$io->section($this->user->lang('CLI_THUMBNAIL_GENERATING')); + +		$sql = 'SELECT COUNT(*) AS nb_missing_thumbnails +			FROM ' . ATTACHMENTS_TABLE . ' +			WHERE thumbnail = 0'; +		$result = $this->db->sql_query($sql); +		$nb_missing_thumbnails = (int) $this->db->sql_fetchfield('nb_missing_thumbnails'); +		$this->db->sql_freeresult($result); + +		if ($nb_missing_thumbnails === 0) +		{ +			$io->warning($this->user->lang('CLI_THUMBNAIL_NOTHING_TO_GENERATE')); +			return 0; +		} + +		$extensions = $this->cache->obtain_attach_extensions(true); + +		$sql = 'SELECT attach_id, physical_filename, extension, real_filename, mimetype +			FROM ' . ATTACHMENTS_TABLE . ' +			WHERE thumbnail = 0'; +		$result = $this->db->sql_query($sql); + +		if (!function_exists('create_thumbnail')) +		{ +			require($this->phpbb_root_path . 'includes/functions_posting.' . $this->php_ext); +		} + +		$progress = $io->createProgressBar($nb_missing_thumbnails); +		if ($output->getVerbosity() === OutputInterface::VERBOSITY_VERBOSE) +		{ +			$progress->setFormat('<info>[%percent:3s%%]</info> %message%'); +			$progress->setOverwrite(false); +		} +		else if ($output->getVerbosity() >= OutputInterface::VERBOSITY_VERY_VERBOSE) +		{ +			$progress->setFormat('<info>[%current:s%/%max:s%]</info><comment>[%elapsed%/%estimated%][%memory%]</comment> %message%'); +			$progress->setOverwrite(false); +		} +		else +		{ +			$io->newLine(2); +			$progress->setFormat( +				"    %current:s%/%max:s% %bar%  %percent:3s%%\n" . +				"                         %elapsed:6s%/%estimated:-6s% %memory:6s%\n"); +			$progress->setBarWidth(60); +		} + +		if (!defined('PHP_WINDOWS_VERSION_BUILD')) +		{ +			$progress->setEmptyBarCharacter('░'); // light shade character \u2591 +			$progress->setProgressCharacter(''); +			$progress->setBarCharacter('▓'); // dark shade character \u2593 +		} + +		$progress->setMessage($this->user->lang('CLI_THUMBNAIL_GENERATING')); + +		$progress->start(); + +		$thumbnail_created = array(); +		while ($row = $this->db->sql_fetchrow($result)) +		{ +			if (isset($extensions[$row['extension']]['display_cat']) && $extensions[$row['extension']]['display_cat'] == ATTACHMENT_CATEGORY_IMAGE) +			{ +				$source = $this->phpbb_root_path . 'files/' . $row['physical_filename']; +				$destination = $this->phpbb_root_path . 'files/thumb_' . $row['physical_filename']; + +				if (create_thumbnail($source, $destination, $row['mimetype'])) +				{ +					$thumbnail_created[] = (int) $row['attach_id']; + +					if (count($thumbnail_created) === 250) +					{ +						$this->commit_changes($thumbnail_created); +						$thumbnail_created = array(); +					} + +					$progress->setMessage($this->user->lang('CLI_THUMBNAIL_GENERATED', $row['real_filename'], $row['physical_filename'])); +				} +				else +				{ +					$progress->setMessage('<info>' . $this->user->lang('CLI_THUMBNAIL_SKIPPED', $row['real_filename'], $row['physical_filename']) . '</info>'); +				} +			} + +			$progress->advance(); +		} +		$this->db->sql_freeresult($result); + +		if (!empty($thumbnail_created)) +		{ +			$this->commit_changes($thumbnail_created); +		} + +		$progress->finish(); + +		$io->newLine(2); +		$io->success($this->user->lang('CLI_THUMBNAIL_GENERATING_DONE')); + +		return 0; +	} + +	/** +	* Commits the changes to the database +	* +	* @param array $thumbnail_created +	*/ +	protected function commit_changes(array $thumbnail_created) +	{ +		$sql = 'UPDATE ' . ATTACHMENTS_TABLE . ' +				SET thumbnail = 1 +				WHERE ' . $this->db->sql_in_set('attach_id', $thumbnail_created); +		$this->db->sql_query($sql); +	} +} diff --git a/phpBB/phpbb/console/command/thumbnail/recreate.php b/phpBB/phpbb/console/command/thumbnail/recreate.php new file mode 100644 index 0000000000..382da290bf --- /dev/null +++ b/phpBB/phpbb/console/command/thumbnail/recreate.php @@ -0,0 +1,72 @@ +<?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\console\command\thumbnail; + +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\ArrayInput; +use Symfony\Component\Console\Output\OutputInterface; + +class recreate extends \phpbb\console\command\command +{ +	/** +	* Sets the command name and description +	* +	* @return null +	*/ +	protected function configure() +	{ +		$this +			->setName('thumbnail:recreate') +			->setDescription($this->user->lang('CLI_DESCRIPTION_THUMBNAIL_RECREATE')) +		; +	} + +	/** +	* Executes the command thumbnail:recreate. +	* +	* This command is a "macro" to execute thumbnail:delete and then thumbnail:generate. +	* +	* @param InputInterface $input The input stream used to get the argument and verboe option. +	* @param OutputInterface $output The output stream, used for printing verbose-mode and error information. +	* +	* @return int 0 if all is ok, 1 if a thumbnail couldn't be deleted. +	*/ +	protected function execute(InputInterface $input, OutputInterface $output) +	{ +		$parameters = array( +			'command' => 'thumbnail:delete' +		); + +		if ($input->getOption('verbose')) +		{ +			$parameters['-' . str_repeat('v', $output->getVerbosity() - 1)] = true; +		} + +		$this->getApplication()->setAutoExit(false); + +		$input_delete = new ArrayInput($parameters); +		$return = $this->getApplication()->run($input_delete, $output); + +		if ($return === 0) +		{ +			$parameters['command'] = 'thumbnail:generate'; + +			$input_create = new ArrayInput($parameters); +			$return = $this->getApplication()->run($input_create, $output); +		} + +		$this->getApplication()->setAutoExit(true); + +		return $return; +	} +} diff --git a/phpBB/phpbb/console/exception_subscriber.php b/phpBB/phpbb/console/exception_subscriber.php new file mode 100644 index 0000000000..b920d4abae --- /dev/null +++ b/phpBB/phpbb/console/exception_subscriber.php @@ -0,0 +1,74 @@ +<?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\console; + +use phpbb\exception\exception_interface; +use Symfony\Component\Console\ConsoleEvents; +use Symfony\Component\Console\Event\ConsoleExceptionEvent; +use Symfony\Component\EventDispatcher\EventSubscriberInterface; + +class exception_subscriber implements EventSubscriberInterface +{ +	/** +	 * @var \phpbb\language\language +	 */ +	protected $language; + +	/** +	 * Construct method +	 * +	 * @param \phpbb\language\language $language Language object +	 * @param bool $debug Debug mode +	 */ +	public function __construct(\phpbb\language\language $language, $debug = false) +	{ +		$this->language = $language; +		$this->debug = $debug; +	} + +	/** +	 * This listener is run when the ConsoleEvents::EXCEPTION event is triggered. +	 * It translate the exception message. If din debug mode the original exception is embedded. +	 * +	 * @param ConsoleExceptionEvent $event +	 */ +	public function on_exception(ConsoleExceptionEvent $event) +	{ +		$original_exception = $event->getException(); + +		if ($original_exception instanceof exception_interface) +		{ +			$parameters = array_merge(array($original_exception->getMessage()), $original_exception->get_parameters()); +			$message = call_user_func_array(array($this->language, 'lang'), $parameters); + +			if ($this->debug) +			{ +				$exception = new \RuntimeException($message , $original_exception->getCode(), $original_exception); +			} +			else +			{ +				$exception = new \RuntimeException($message , $original_exception->getCode()); +			} + +			$event->setException($exception); +		} +	} + +	static public function getSubscribedEvents() +	{ +		return array( +			ConsoleEvents::EXCEPTION => 'on_exception', +		); +	} +} | 
