PearRepository.php 8.1 KB

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