PearRepository.php 8.2 KB

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