PearRepository.php 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. <?php
  2. /*
  3. * This file is part of Composer.
  4. *
  5. * (c) Nils Adermann <naderman@naderman.de>
  6. * Jordi Boggiano <j.boggiano@seld.be>
  7. *
  8. * For the full copyright and license information, please view the LICENSE
  9. * file that was distributed with this source code.
  10. */
  11. namespace Composer\Repository;
  12. use Composer\IO\IOInterface;
  13. use Composer\Semver\VersionParser as SemverVersionParser;
  14. use Composer\Package\Version\VersionParser;
  15. use Composer\Repository\Pear\ChannelReader;
  16. use Composer\Package\CompletePackage;
  17. use Composer\Repository\Pear\ChannelInfo;
  18. use Composer\EventDispatcher\EventDispatcher;
  19. use Composer\Package\Link;
  20. use Composer\Semver\Constraint\Constraint;
  21. use Composer\Util\RemoteFilesystem;
  22. use Composer\Config;
  23. use Composer\Factory;
  24. /**
  25. * Builds list of package from PEAR channel.
  26. *
  27. * Packages read from channel are named as 'pear-{channelName}/{packageName}'
  28. * and has aliased as 'pear-{channelAlias}/{packageName}'
  29. *
  30. * @author Benjamin Eberlei <kontakt@beberlei.de>
  31. * @author Jordi Boggiano <j.boggiano@seld.be>
  32. */
  33. class PearRepository extends ArrayRepository implements ConfigurableRepositoryInterface
  34. {
  35. private $url;
  36. private $io;
  37. private $rfs;
  38. private $versionParser;
  39. private $repoConfig;
  40. /** @var string vendor makes additional alias for each channel as {prefix}/{packagename}. It allows smoother
  41. * package transition to composer-like repositories.
  42. */
  43. private $vendorAlias;
  44. public function __construct(array $repoConfig, IOInterface $io, Config $config, EventDispatcher $dispatcher = null, RemoteFilesystem $rfs = null)
  45. {
  46. parent::__construct();
  47. if (!preg_match('{^https?://}', $repoConfig['url'])) {
  48. $repoConfig['url'] = 'http://'.$repoConfig['url'];
  49. }
  50. $urlBits = parse_url($repoConfig['url']);
  51. if (empty($urlBits['scheme']) || empty($urlBits['host'])) {
  52. throw new \UnexpectedValueException('Invalid url given for PEAR repository: '.$repoConfig['url']);
  53. }
  54. $this->url = rtrim($repoConfig['url'], '/');
  55. $this->io = $io;
  56. $this->rfs = $rfs ?: Factory::createRemoteFilesystem($this->io, $config);
  57. $this->vendorAlias = isset($repoConfig['vendor-alias']) ? $repoConfig['vendor-alias'] : null;
  58. $this->versionParser = new VersionParser();
  59. $this->repoConfig = $repoConfig;
  60. }
  61. public function getRepoConfig()
  62. {
  63. return $this->repoConfig;
  64. }
  65. protected function initialize()
  66. {
  67. parent::initialize();
  68. $this->io->writeError('Initializing PEAR repository '.$this->url);
  69. $reader = new ChannelReader($this->rfs);
  70. try {
  71. $channelInfo = $reader->read($this->url);
  72. } catch (\Exception $e) {
  73. $this->io->writeError('<warning>PEAR repository from '.$this->url.' could not be loaded. '.$e->getMessage().'</warning>');
  74. return;
  75. }
  76. $packages = $this->buildComposerPackages($channelInfo, $this->versionParser);
  77. foreach ($packages as $package) {
  78. $this->addPackage($package);
  79. }
  80. }
  81. /**
  82. * Builds CompletePackages from PEAR package definition data.
  83. *
  84. * @param ChannelInfo $channelInfo
  85. * @param SemverVersionParser $versionParser
  86. * @return CompletePackage
  87. */
  88. private function buildComposerPackages(ChannelInfo $channelInfo, SemverVersionParser $versionParser)
  89. {
  90. $result = array();
  91. foreach ($channelInfo->getPackages() as $packageDefinition) {
  92. foreach ($packageDefinition->getReleases() as $version => $releaseInfo) {
  93. try {
  94. $normalizedVersion = $versionParser->normalize($version);
  95. } catch (\UnexpectedValueException $e) {
  96. $this->io->writeError('Could not load '.$packageDefinition->getPackageName().' '.$version.': '.$e->getMessage(), true, IOInterface::VERBOSE);
  97. continue;
  98. }
  99. $composerPackageName = $this->buildComposerPackageName($packageDefinition->getChannelName(), $packageDefinition->getPackageName());
  100. // distribution url must be read from /r/{packageName}/{version}.xml::/r/g:text()
  101. // but this location is 'de-facto' standard
  102. $urlBits = parse_url($this->url);
  103. $scheme = (isset($urlBits['scheme']) && 'https' === $urlBits['scheme'] && extension_loaded('openssl')) ? 'https' : 'http';
  104. $distUrl = "{$scheme}://{$packageDefinition->getChannelName()}/get/{$packageDefinition->getPackageName()}-{$version}.tgz";
  105. $requires = array();
  106. $suggests = array();
  107. $conflicts = array();
  108. $replaces = array();
  109. // alias package only when its channel matches repository channel,
  110. // cause we've know only repository channel alias
  111. if ($channelInfo->getName() == $packageDefinition->getChannelName()) {
  112. $composerPackageAlias = $this->buildComposerPackageName($channelInfo->getAlias(), $packageDefinition->getPackageName());
  113. $aliasConstraint = new Constraint('==', $normalizedVersion);
  114. $replaces[] = new Link($composerPackageName, $composerPackageAlias, $aliasConstraint, 'replaces', (string) $aliasConstraint);
  115. }
  116. // alias package with user-specified prefix. it makes private pear channels looks like composer's.
  117. if (!empty($this->vendorAlias)
  118. && ($this->vendorAlias != 'pear-'.$channelInfo->getAlias() || $channelInfo->getName() != $packageDefinition->getChannelName())
  119. ) {
  120. $composerPackageAlias = "{$this->vendorAlias}/{$packageDefinition->getPackageName()}";
  121. $aliasConstraint = new Constraint('==', $normalizedVersion);
  122. $replaces[] = new Link($composerPackageName, $composerPackageAlias, $aliasConstraint, 'replaces', (string) $aliasConstraint);
  123. }
  124. foreach ($releaseInfo->getDependencyInfo()->getRequires() as $dependencyConstraint) {
  125. $dependencyPackageName = $this->buildComposerPackageName($dependencyConstraint->getChannelName(), $dependencyConstraint->getPackageName());
  126. $constraint = $versionParser->parseConstraints($dependencyConstraint->getConstraint());
  127. $link = new Link($composerPackageName, $dependencyPackageName, $constraint, $dependencyConstraint->getType(), $dependencyConstraint->getConstraint());
  128. switch ($dependencyConstraint->getType()) {
  129. case 'required':
  130. $requires[] = $link;
  131. break;
  132. case 'conflicts':
  133. $conflicts[] = $link;
  134. break;
  135. case 'replaces':
  136. $replaces[] = $link;
  137. break;
  138. }
  139. }
  140. foreach ($releaseInfo->getDependencyInfo()->getOptionals() as $group => $dependencyConstraints) {
  141. foreach ($dependencyConstraints as $dependencyConstraint) {
  142. $dependencyPackageName = $this->buildComposerPackageName($dependencyConstraint->getChannelName(), $dependencyConstraint->getPackageName());
  143. $suggests[$group.'-'.$dependencyPackageName] = $dependencyConstraint->getConstraint();
  144. }
  145. }
  146. $package = new CompletePackage($composerPackageName, $normalizedVersion, $version);
  147. $package->setType('pear-library');
  148. $package->setDescription($packageDefinition->getDescription());
  149. $package->setLicense(array($packageDefinition->getLicense()));
  150. $package->setDistType('file');
  151. $package->setDistUrl($distUrl);
  152. $package->setAutoload(array('classmap' => array('')));
  153. $package->setIncludePaths(array('/'));
  154. $package->setRequires($requires);
  155. $package->setConflicts($conflicts);
  156. $package->setSuggests($suggests);
  157. $package->setReplaces($replaces);
  158. $result[] = $package;
  159. }
  160. }
  161. return $result;
  162. }
  163. private function buildComposerPackageName($channelName, $packageName)
  164. {
  165. if ('php' === $channelName) {
  166. return "php";
  167. }
  168. if ('ext' === $channelName) {
  169. return "ext-{$packageName}";
  170. }
  171. return "pear-{$channelName}/{$packageName}";
  172. }
  173. }