diff --git a/phpBB/adm/style/acp_ext_gallery.html b/phpBB/adm/style/acp_ext_gallery.html new file mode 100644 index 0000000000..9be59429f6 --- /dev/null +++ b/phpBB/adm/style/acp_ext_gallery.html @@ -0,0 +1,57 @@ + + + + +

{{lang( 'EXTENSIONS_ADMIN') }}

+ +

{{lang( 'EXTENSIONS_EXPLAIN') }}

+ +
+ {{ lang('BROWSE_EXTENSIONS_DATABASE') }}{{ lang('SETTINGS') }} +
+ + + + + + + + + + + + + + {% for extension in extensions %} + + + + + + {% endfor %} + +
{{ lang("EXTENSION_NAME") }}{{ lang("VERSION") }}{{ lang("DESCRIPTION") }}
+ {{ extension.name }}
+ {{ lang('DETAILS') }}{{ lang('INSTALL') }} +
{{ extension.version }}{{ extension.description }}
+ + diff --git a/phpBB/includes/acp/acp_extensions.php b/phpBB/includes/acp/acp_extensions.php index c08674841c..01e0b5e245 100644 --- a/phpBB/includes/acp/acp_extensions.php +++ b/phpBB/includes/acp/acp_extensions.php @@ -28,10 +28,13 @@ class acp_extensions var $tpl_name; var $page_title; + private $db; private $config; private $template; private $user; private $log; + + /** @var \phpbb\request\request */ private $request; private $phpbb_dispatcher; private $ext_manager; @@ -39,8 +42,10 @@ class acp_extensions function main() { // Start the page - global $config, $user, $template, $request, $phpbb_extension_manager, $phpbb_root_path, $phpbb_log, $phpbb_dispatcher; + global $config, $user, $template, $request, $phpbb_extension_manager, $db, $phpbb_root_path, $phpbb_log, $phpbb_dispatcher; + + $this->db = $db; $this->config = $config; $this->template = $template; $this->user = $user; @@ -49,7 +54,22 @@ class acp_extensions $this->phpbb_dispatcher = $phpbb_dispatcher; $this->ext_manager = $phpbb_extension_manager; - $this->user->add_lang(array('install', 'acp/extensions', 'migrator')); + $this->user->add_lang(['install', 'acp/extensions', 'migrator']); + + switch ($mode) + { + case 'gallery': + $this->gallery_mode(); + break; + default: + $this->main_mode(); + break; + } + } + + public function main_mode() + { + global $phpbb_extension_manager, $phpbb_root_path; $this->page_title = 'ACP_EXTENSIONS'; @@ -381,6 +401,20 @@ class acp_extensions $this->tpl_name = $tpl_name; } + public function gallery_mode() + { + global $phpbb_container; + + /** @var \phpbb\composer\extension_manager $manager */ + $manager = $phpbb_container->get('ext.composer.manager'); + $this->page_title = 'ACP_EXTENSIONS_GALLERY'; + $this->tpl_name = 'acp_ext_gallery'; + + $this->request->enable_super_globals(); + $this->template->assign_var('extensions', $manager->get_available_packages()); + $this->request->disable_super_globals(); + } + /** * Lists all the enabled extensions and dumps to the template * diff --git a/phpBB/includes/acp/info/acp_extensions.php b/phpBB/includes/acp/info/acp_extensions.php index 9adcd543b9..5d9e420fda 100644 --- a/phpBB/includes/acp/info/acp_extensions.php +++ b/phpBB/includes/acp/info/acp_extensions.php @@ -20,6 +20,7 @@ class acp_extensions_info 'title' => 'ACP_EXTENSION_MANAGEMENT', 'modes' => array( 'main' => array('title' => 'ACP_EXTENSIONS', 'auth' => 'acl_a_extensions', 'cat' => array('ACP_EXTENSION_MANAGEMENT')), + 'gallery' => array('title' => 'ACP_EXTENSIONS_GALLERY', 'auth' => 'acl_a_extensions', 'cat' => array('ACP_EXTENSION_MANAGEMENT')), ), ); } diff --git a/phpBB/language/en/acp/common.php b/phpBB/language/en/acp/common.php index 7f341abd07..40b4a5d9d4 100644 --- a/phpBB/language/en/acp/common.php +++ b/phpBB/language/en/acp/common.php @@ -88,6 +88,7 @@ $lang = array_merge($lang, array( 'ACP_EXTENSION_GROUPS' => 'Manage attachment extension groups', 'ACP_EXTENSION_MANAGEMENT' => 'Extension management', 'ACP_EXTENSIONS' => 'Manage extensions', + 'ACP_EXTENSIONS_GALLERY' => 'Extensions gallery', 'ACP_FORUM_BASED_PERMISSIONS' => 'Forum based permissions', 'ACP_FORUM_LOGS' => 'Forum logs', diff --git a/phpBB/phpbb/composer/installer.php b/phpBB/phpbb/composer/installer.php index 73d90c6658..1135e13c8e 100644 --- a/phpBB/phpbb/composer/installer.php +++ b/phpBB/phpbb/composer/installer.php @@ -19,6 +19,8 @@ use Composer\IO\IOInterface; use Composer\IO\NullIO; use Composer\Json\JsonFile; use Composer\Package\CompletePackage; +use Composer\Package\LinkConstraint\LinkConstraintInterface; +use Composer\Package\LinkConstraint\VersionConstraint; use Composer\Package\PackageInterface; use Composer\Repository\ComposerRepository; use Composer\Repository\RepositoryInterface; @@ -198,62 +200,101 @@ class installer $composer = Factory::create($io, $this->get_composer_ext_json_filename(), false); + /** @var LinkConstraintInterface $core_constraint */ + $core_constraint = $composer->getPackage()->getRequires()['phpbb/phpbb']->getConstraint(); + $available = []; + + $compatible_packages = []; $repositories = $composer->getRepositoryManager()->getRepositories(); /** @var RepositoryInterface $repository */ foreach ($repositories as $repository) { - if ($repository instanceof ComposerRepository && $repository->hasProviders()) + try { - $r = new \ReflectionObject($repository); - $repo_url = $r->getProperty('url'); - $repo_url->setAccessible(true); - - if ($repo_url->getValue($repository) === 'http://packagist.org') + if ($repository instanceof ComposerRepository && $repository->hasProviders()) { - $url = 'https://packagist.org/packages/list.json?type=' . $type; - $rfs = new RemoteFilesystem($io); - $hostname = parse_url($url, PHP_URL_HOST) ?: $url; - $json = $rfs->getContents($hostname, $url, false); + // Special case for packagist which exposes an api to retrieve all packages of a given type. + // For the others composer repositories with providers we can't do anything. It would be too slow. - /** @var PackageInterface $package */ - foreach (JsonFile::parseJson($json, $url)['packageNames'] as $package) + $r = new \ReflectionObject($repository); + $repo_url = $r->getProperty('url'); + $repo_url->setAccessible(true); + + if ($repo_url->getValue($repository) === 'http://packagist.org') { - $packages = $repository->findPackages($package); - $package = array_pop($packages); + $url = 'https://packagist.org/packages/list.json?type=' . $type; + $rfs = new RemoteFilesystem($io); + $hostname = parse_url($url, PHP_URL_HOST) ?: $url; + $json = $rfs->getContents($hostname, $url, false); - if (isset($available[$package->getName()])) + /** @var PackageInterface $package */ + foreach (JsonFile::parseJson($json, $url)['packageNames'] as $package) { - continue; + $versions = $repository->findPackages($package); + $compatible_packages = $this->get_compatible_versions($compatible_packages, $core_constraint, $package, $versions); } - - $available[$package->getName()] = ['name' => $package->getPrettyName()]; - - if ($package instanceof CompletePackage) + } + } + else + { + // Pre-filter repo packages by their type + $packages = []; + /** @var PackageInterface $package */ + foreach ($repository->getPackages() as $package) + { + if ($package->getType() === $type) { - $available[$package->getName()]['description'] = $package->getDescription(); - $available[$package->getName()]['url'] = $package->getHomepage(); + $packages[$package->getName()][] = $package; } } + + // Filter the compatibles versions + foreach ($packages as $package => $versions) + { + $compatible_packages = $this->get_compatible_versions($compatible_packages, $core_constraint, $package, $versions); + } } } + catch (\Exception $e) + { + // If a repo fails, just skip it. + continue; + } + } + + foreach ($compatible_packages as $name => $versions) + { + // Determine the highest version of the package + /** @var CompletePackage $highest_version */ + $highest_version = null; + + /** @var CompletePackage $version */ + foreach ($versions as $version) + { + if (!$highest_version || version_compare($version->getVersion(), $highest_version->getVersion(), '>')) + { + $highest_version = $version; + } + } + + // Generates the entry + $available[$name] = []; + $available[$name]['name'] = $highest_version->getPrettyName(); + $available[$name]['version'] = $highest_version->getPrettyVersion(); + + if ($version instanceof CompletePackage) + { + $available[$name]['description'] = $highest_version->getDescription(); + $available[$name]['url'] = $highest_version->getHomepage(); + $available[$name]['authors'] = $highest_version->getAuthors(); + } else { - /** @var PackageInterface $package */ - foreach ($repository->getPackages() as $package) - { - if ($package->getType() === $type) - { - $available[$package->getName()] = ['name' => $package->getPrettyName()]; - - if ($package instanceof CompletePackage) - { - $available[$package->getName()]['description'] = $package->getDescription(); - $available[$package->getName()]['url'] = $package->getHomepage(); - } - } - } + $available[$name]['description'] = ''; + $available[$name]['url'] = ''; + $available[$name]['authors'] = []; } } @@ -267,6 +308,38 @@ class installer } } + /** + * Updates $compatible_packages with the versions of $versions compatibles with the $core_constraint + * + * @param array $compatible_packages List of compatibles versions + * @param LinkConstraintInterface $core_constraint Constraint against the phpBB version + * @param string $package_name Considered package + * @param array $versions List of available versions + * + * @return array + */ + private function get_compatible_versions(array $compatible_packages, LinkConstraintInterface $core_constraint, $package_name, array $versions) + { + /** @var PackageInterface $version */ + foreach ($versions as $version) + { + if (array_key_exists('phpbb/phpbb', $version->getRequires())) + { + /** @var LinkConstraintInterface $package_constraint */ + $package_constraint = $version->getRequires()['phpbb/phpbb']->getConstraint(); + + if (!$package_constraint->matches($core_constraint)) + { + continue; + } + } + + $compatible_packages[$package_name][] = $version; + } + + return $compatible_packages; + } + /** * Generates and write the json file used to install the set of packages * @@ -276,7 +349,7 @@ class installer protected function generate_ext_json_file(array $packages) { $io = new NullIO(); - $composer = Factory::create($io, null, false); + $composer = Factory::create($io, $this->root_path . 'composer.json', false); $core_packages = $this->get_core_packages($composer); $core_json_data = [ @@ -314,6 +387,27 @@ class installer return $core_deps; } + /** + * Get the core installed packages + * + * @param Composer $composer Composer object to load the dependencies + * @return array The core packages with their version + */ + protected function get_core_version(Composer $composer) + { + $core_deps = []; + $packages = $composer->getRepositoryManager()->getLocalRepository()->getCanonicalPackages(); + + foreach ($packages as $package) + { + $core_deps[$package->getName()] = $package->getPrettyVersion(); + } + + $core_deps['phpbb/phpbb'] = $composer->getPackage()->getPrettyVersion(); + + return $core_deps; + } + /** * Get the PHP version required by the core * diff --git a/phpBB/phpbb/composer/manager.php b/phpBB/phpbb/composer/manager.php index f05fec4957..5f7f2e28d9 100644 --- a/phpBB/phpbb/composer/manager.php +++ b/phpBB/phpbb/composer/manager.php @@ -187,7 +187,7 @@ class manager implements manager_interface if ($this->available_packages === null) { $this->available_packages = $this->cache->get('_composer_' . $this->package_type . '_available'); - if ($this->available_packages === false) + if (!$this->available_packages) { $this->available_packages = $this->installer->get_available_packages($this->package_type); $this->cache->put('_composer_' . $this->package_type . '_available', $this->available_packages, 24*60*60); diff --git a/phpBB/phpbb/console/command/extension/list_available.php b/phpBB/phpbb/console/command/extension/list_available.php index 107aca3410..c844201fd3 100644 --- a/phpBB/phpbb/console/command/extension/list_available.php +++ b/phpBB/phpbb/console/command/extension/list_available.php @@ -66,7 +66,8 @@ class list_available extends \phpbb\console\command\command foreach ($this->manager->get_available_packages() as $package) { - $extensions[] = '' . $package['name'] . ' ' . $package['url'] . "\n" . $package['description']; + $extensions[] = '' . $package['name'] . ' (' . $package['version'] . ') ' . $package['url'] . + ($package['description'] ? "\n" . $package['description'] : ''); } $io->listing($extensions); diff --git a/phpBB/phpbb/db/migration/data/v320/extensions_composer.php b/phpBB/phpbb/db/migration/data/v320/extensions_composer.php index 09771f1797..1271adb358 100644 --- a/phpBB/phpbb/db/migration/data/v320/extensions_composer.php +++ b/phpBB/phpbb/db/migration/data/v320/extensions_composer.php @@ -22,6 +22,14 @@ class extensions_composer extends \phpbb\db\migration\migration array('config.add', array('exts_composer_packagist', true)), array('config.add', array('exts_composer_json_file', 'composer-ext.json')), array('config.add', array('exts_composer_vendor_dir', 'vendor-ext/')), + array('module.add', array( + 'acp', + 'ACP_EXTENSION_MANAGEMENT', + array( + 'module_basename' => 'acp_extensions', + 'modes' => array('gallery'), + ), + )), ); } }