123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444 |
- <?php
- /*
- * This file is part of Composer.
- *
- * (c) Nils Adermann <naderman@naderman.de>
- * Jordi Boggiano <j.boggiano@seld.be>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Composer\Repository;
- use Composer\Downloader\TransportException;
- use Composer\Repository\Vcs\VcsDriverInterface;
- use Composer\Package\Version\VersionParser;
- use Composer\Package\Loader\ArrayLoader;
- use Composer\Package\Loader\ValidatingArrayLoader;
- use Composer\Package\Loader\InvalidPackageException;
- use Composer\Package\Loader\LoaderInterface;
- use Composer\EventDispatcher\EventDispatcher;
- use Composer\Util\ProcessExecutor;
- use Composer\Util\HttpDownloader;
- use Composer\Util\Url;
- use Composer\Semver\Constraint\Constraint;
- use Composer\IO\IOInterface;
- use Composer\Config;
- /**
- * @author Jordi Boggiano <j.boggiano@seld.be>
- */
- class VcsRepository extends ArrayRepository implements ConfigurableRepositoryInterface
- {
- protected $url;
- protected $packageName;
- protected $isVerbose;
- protected $isVeryVerbose;
- protected $io;
- protected $config;
- protected $versionParser;
- protected $type;
- protected $loader;
- protected $repoConfig;
- protected $httpDownloader;
- protected $processExecutor;
- protected $branchErrorOccurred = false;
- private $drivers;
- /** @var VcsDriverInterface */
- private $driver;
- /** @var VersionCacheInterface */
- private $versionCache;
- private $emptyReferences = array();
- private $versionTransportExceptions = array();
- public function __construct(array $repoConfig, IOInterface $io, Config $config, HttpDownloader $httpDownloader, EventDispatcher $dispatcher = null, array $drivers = null, VersionCacheInterface $versionCache = null)
- {
- parent::__construct();
- $this->drivers = $drivers ?: array(
- 'github' => 'Composer\Repository\Vcs\GitHubDriver',
- 'gitlab' => 'Composer\Repository\Vcs\GitLabDriver',
- 'git-bitbucket' => 'Composer\Repository\Vcs\GitBitbucketDriver',
- 'git' => 'Composer\Repository\Vcs\GitDriver',
- 'hg-bitbucket' => 'Composer\Repository\Vcs\HgBitbucketDriver',
- 'hg' => 'Composer\Repository\Vcs\HgDriver',
- 'perforce' => 'Composer\Repository\Vcs\PerforceDriver',
- 'fossil' => 'Composer\Repository\Vcs\FossilDriver',
- // svn must be last because identifying a subversion server for sure is practically impossible
- 'svn' => 'Composer\Repository\Vcs\SvnDriver',
- );
- $this->url = $repoConfig['url'];
- $this->io = $io;
- $this->type = isset($repoConfig['type']) ? $repoConfig['type'] : 'vcs';
- $this->isVerbose = $io->isVerbose();
- $this->isVeryVerbose = $io->isVeryVerbose();
- $this->config = $config;
- $this->repoConfig = $repoConfig;
- $this->versionCache = $versionCache;
- $this->httpDownloader = $httpDownloader;
- $this->processExecutor = new ProcessExecutor($io);
- }
- public function getRepoName()
- {
- $driverClass = get_class($this->getDriver());
- $driverType = array_search($driverClass, $this->drivers);
- if (!$driverType) {
- $driverType = $driverClass;
- }
- return 'vcs repo ('.$driverType.' '.Url::sanitize($this->url).')';
- }
- public function getRepoConfig()
- {
- return $this->repoConfig;
- }
- public function setLoader(LoaderInterface $loader)
- {
- $this->loader = $loader;
- }
- public function getDriver()
- {
- if ($this->driver) {
- return $this->driver;
- }
- if (isset($this->drivers[$this->type])) {
- $class = $this->drivers[$this->type];
- $this->driver = new $class($this->repoConfig, $this->io, $this->config, $this->httpDownloader, $this->processExecutor);
- $this->driver->initialize();
- return $this->driver;
- }
- foreach ($this->drivers as $driver) {
- if ($driver::supports($this->io, $this->config, $this->url)) {
- $this->driver = new $driver($this->repoConfig, $this->io, $this->config, $this->httpDownloader, $this->processExecutor);
- $this->driver->initialize();
- return $this->driver;
- }
- }
- foreach ($this->drivers as $driver) {
- if ($driver::supports($this->io, $this->config, $this->url, true)) {
- $this->driver = new $driver($this->repoConfig, $this->io, $this->config, $this->httpDownloader, $this->processExecutor);
- $this->driver->initialize();
- return $this->driver;
- }
- }
- }
- public function hadInvalidBranches()
- {
- return $this->branchErrorOccurred;
- }
- public function getEmptyReferences()
- {
- return $this->emptyReferences;
- }
- public function getVersionTransportExceptions()
- {
- return $this->versionTransportExceptions;
- }
- protected function initialize()
- {
- parent::initialize();
- $isVerbose = $this->isVerbose;
- $isVeryVerbose = $this->isVeryVerbose;
- $driver = $this->getDriver();
- if (!$driver) {
- throw new \InvalidArgumentException('No driver found to handle VCS repository '.$this->url);
- }
- $this->versionParser = new VersionParser;
- if (!$this->loader) {
- $this->loader = new ArrayLoader($this->versionParser);
- }
- try {
- if ($driver->hasComposerFile($driver->getRootIdentifier())) {
- $data = $driver->getComposerInformation($driver->getRootIdentifier());
- $this->packageName = !empty($data['name']) ? $data['name'] : null;
- }
- } catch (\Exception $e) {
- if ($isVeryVerbose) {
- $this->io->writeError('<error>Skipped parsing '.$driver->getRootIdentifier().', '.$e->getMessage().'</error>');
- }
- }
- foreach ($driver->getTags() as $tag => $identifier) {
- $msg = 'Reading composer.json of <info>' . ($this->packageName ?: $this->url) . '</info> (<comment>' . $tag . '</comment>)';
- if ($isVeryVerbose) {
- $this->io->writeError($msg);
- } elseif ($isVerbose) {
- $this->io->overwriteError($msg, false);
- }
- // strip the release- prefix from tags if present
- $tag = str_replace('release-', '', $tag);
- $cachedPackage = $this->getCachedPackageVersion($tag, $identifier, $isVerbose, $isVeryVerbose);
- if ($cachedPackage) {
- $this->addPackage($cachedPackage);
- continue;
- } elseif ($cachedPackage === false) {
- $this->emptyReferences[] = $identifier;
- continue;
- }
- if (!$parsedTag = $this->validateTag($tag)) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped tag '.$tag.', invalid tag name</warning>');
- }
- continue;
- }
- try {
- if (!$data = $driver->getComposerInformation($identifier)) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped tag '.$tag.', no composer file</warning>');
- }
- $this->emptyReferences[] = $identifier;
- continue;
- }
- // manually versioned package
- if (isset($data['version'])) {
- $data['version_normalized'] = $this->versionParser->normalize($data['version']);
- } else {
- // auto-versioned package, read value from tag
- $data['version'] = $tag;
- $data['version_normalized'] = $parsedTag;
- }
- // make sure tag packages have no -dev flag
- $data['version'] = preg_replace('{[.-]?dev$}i', '', $data['version']);
- $data['version_normalized'] = preg_replace('{(^dev-|[.-]?dev$)}i', '', $data['version_normalized']);
- // broken package, version doesn't match tag
- if ($data['version_normalized'] !== $parsedTag) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped tag '.$tag.', tag ('.$parsedTag.') does not match version ('.$data['version_normalized'].') in composer.json</warning>');
- }
- continue;
- }
- $tagPackageName = isset($data['name']) ? $data['name'] : $this->packageName;
- if ($existingPackage = $this->findPackage($tagPackageName, $data['version_normalized'])) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped tag '.$tag.', it conflicts with an another tag ('.$existingPackage->getPrettyVersion().') as both resolve to '.$data['version_normalized'].' internally</warning>');
- }
- continue;
- }
- if ($isVeryVerbose) {
- $this->io->writeError('Importing tag '.$tag.' ('.$data['version_normalized'].')');
- }
- $this->addPackage($this->loader->load($this->preProcess($driver, $data, $identifier)));
- } catch (\Exception $e) {
- if ($e instanceof TransportException) {
- $this->versionTransportExceptions['tags'][$tag] = $e;
- if ($e->getCode() === 404) {
- $this->emptyReferences[] = $identifier;
- }
- }
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped tag '.$tag.', '.($e instanceof TransportException ? 'no composer file was found (' . $e->getCode() . ' HTTP status code)' : $e->getMessage()).'</warning>');
- }
- continue;
- }
- }
- if (!$isVeryVerbose) {
- $this->io->overwriteError('', false);
- }
- $branches = $driver->getBranches();
- foreach ($branches as $branch => $identifier) {
- $msg = 'Reading composer.json of <info>' . ($this->packageName ?: $this->url) . '</info> (<comment>' . $branch . '</comment>)';
- if ($isVeryVerbose) {
- $this->io->writeError($msg);
- } elseif ($isVerbose) {
- $this->io->overwriteError($msg, false);
- }
- if ($branch === 'trunk' && isset($branches['master'])) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped branch '.$branch.', can not parse both master and trunk branches as they both resolve to '.VersionParser::DEV_MASTER_ALIAS.' internally</warning>');
- }
- continue;
- }
- if (!$parsedBranch = $this->validateBranch($branch)) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped branch '.$branch.', invalid name</warning>');
- }
- continue;
- }
- // make sure branch packages have a dev flag
- if ('dev-' === substr($parsedBranch, 0, 4) || VersionParser::DEV_MASTER_ALIAS === $parsedBranch) {
- $version = 'dev-' . $branch;
- } else {
- $prefix = substr($branch, 0, 1) === 'v' ? 'v' : '';
- $version = $prefix . preg_replace('{(\.9{7})+}', '.x', $parsedBranch);
- }
- $cachedPackage = $this->getCachedPackageVersion($version, $identifier, $isVerbose, $isVeryVerbose);
- if ($cachedPackage) {
- $this->addPackage($cachedPackage);
- continue;
- } elseif ($cachedPackage === false) {
- $this->emptyReferences[] = $identifier;
- continue;
- }
- try {
- if (!$data = $driver->getComposerInformation($identifier)) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped branch '.$branch.', no composer file</warning>');
- }
- $this->emptyReferences[] = $identifier;
- continue;
- }
- // branches are always auto-versioned, read value from branch name
- $data['version'] = $version;
- $data['version_normalized'] = $parsedBranch;
- if ($isVeryVerbose) {
- $this->io->writeError('Importing branch '.$branch.' ('.$data['version'].')');
- }
- $packageData = $this->preProcess($driver, $data, $identifier);
- $package = $this->loader->load($packageData);
- if ($this->loader instanceof ValidatingArrayLoader && $this->loader->getWarnings()) {
- throw new InvalidPackageException($this->loader->getErrors(), $this->loader->getWarnings(), $packageData);
- }
- $this->addPackage($package);
- } catch (TransportException $e) {
- $this->versionTransportExceptions['branches'][$branch] = $e;
- if ($e->getCode() === 404) {
- $this->emptyReferences[] = $identifier;
- }
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped branch '.$branch.', no composer file was found (' . $e->getCode() . ' HTTP status code)</warning>');
- }
- continue;
- } catch (\Exception $e) {
- if (!$isVeryVerbose) {
- $this->io->writeError('');
- }
- $this->branchErrorOccurred = true;
- $this->io->writeError('<error>Skipped branch '.$branch.', '.$e->getMessage().'</error>');
- $this->io->writeError('');
- continue;
- }
- }
- $driver->cleanup();
- if (!$isVeryVerbose) {
- $this->io->overwriteError('', false);
- }
- if (!$this->getPackages()) {
- throw new InvalidRepositoryException('No valid composer.json was found in any branch or tag of '.$this->url.', could not load a package from it.');
- }
- }
- protected function preProcess(VcsDriverInterface $driver, array $data, $identifier)
- {
- // keep the name of the main identifier for all packages
- $dataPackageName = isset($data['name']) ? $data['name'] : null;
- $data['name'] = $this->packageName ?: $dataPackageName;
- if (!isset($data['dist'])) {
- $data['dist'] = $driver->getDist($identifier);
- }
- if (!isset($data['source'])) {
- $data['source'] = $driver->getSource($identifier);
- }
- return $data;
- }
- private function validateBranch($branch)
- {
- try {
- $normalizedBranch = $this->versionParser->normalizeBranch($branch);
- // validate that the branch name has no weird characters conflicting with constraints
- $this->versionParser->parseConstraints($normalizedBranch);
- return $normalizedBranch;
- } catch (\Exception $e) {
- }
- return false;
- }
- private function validateTag($version)
- {
- try {
- return $this->versionParser->normalize($version);
- } catch (\Exception $e) {
- }
- return false;
- }
- private function getCachedPackageVersion($version, $identifier, $isVerbose, $isVeryVerbose)
- {
- if (!$this->versionCache) {
- return;
- }
- $cachedPackage = $this->versionCache->getVersionPackage($version, $identifier);
- if ($cachedPackage === false) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped '.$version.', no composer file (cached from ref '.$identifier.')</warning>');
- }
- return false;
- }
- if ($cachedPackage) {
- $msg = 'Found cached composer.json of <info>' . ($this->packageName ?: $this->url) . '</info> (<comment>' . $version . '</comment>)';
- if ($isVeryVerbose) {
- $this->io->writeError($msg);
- } elseif ($isVerbose) {
- $this->io->overwriteError($msg, false);
- }
- if ($existingPackage = $this->findPackage($cachedPackage['name'], new Constraint('=', $cachedPackage['version_normalized']))) {
- if ($isVeryVerbose) {
- $this->io->writeError('<warning>Skipped cached version '.$version.', it conflicts with an another tag ('.$existingPackage->getPrettyVersion().') as both resolve to '.$cachedPackage['version_normalized'].' internally</warning>');
- }
- $cachedPackage = null;
- }
- }
- if ($cachedPackage) {
- return $this->loader->load($cachedPackage);
- }
- return null;
- }
- }
|