PearRepository.php 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  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\MemoryPackage;
  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. if (function_exists('filter_var') && version_compare(PHP_VERSION, '5.3.3', '>=') && !filter_var($repoConfig['url'], FILTER_VALIDATE_URL)) {
  46. throw new \UnexpectedValueException('Invalid url given for PEAR repository: '.$repoConfig['url']);
  47. }
  48. $this->url = rtrim($repoConfig['url'], '/');
  49. $this->io = $io;
  50. $this->rfs = $rfs ?: new RemoteFilesystem($this->io);
  51. $this->vendorAlias = isset($repoConfig['vendor-alias']) ? $repoConfig['vendor-alias'] : null;
  52. $this->versionParser = new VersionParser();
  53. }
  54. protected function initialize()
  55. {
  56. parent::initialize();
  57. $this->io->write('Initializing PEAR repository '.$this->url);
  58. $reader = new ChannelReader($this->rfs);
  59. try {
  60. $channelInfo = $reader->read($this->url);
  61. } catch (\Exception $e) {
  62. $this->io->write('<warning>PEAR repository from '.$this->url.' could not be loaded. '.$e->getMessage().'</warning>');
  63. return;
  64. }
  65. $packages = $this->buildComposerPackages($channelInfo, $this->versionParser);
  66. foreach ($packages as $package) {
  67. $this->addPackage($package);
  68. }
  69. }
  70. /**
  71. * Builds MemoryPackages from PEAR package definition data.
  72. *
  73. * @param ChannelInfo $channelInfo
  74. * @return MemoryPackage
  75. */
  76. private function buildComposerPackages(ChannelInfo $channelInfo, VersionParser $versionParser)
  77. {
  78. $result = array();
  79. foreach ($channelInfo->getPackages() as $packageDefinition) {
  80. foreach ($packageDefinition->getReleases() as $version => $releaseInfo) {
  81. $normalizedVersion = $this->parseVersion($version);
  82. if (!$normalizedVersion) {
  83. continue; // skip packages with unparsable versions
  84. }
  85. $composerPackageName = $this->buildComposerPackageName($packageDefinition->getChannelName(), $packageDefinition->getPackageName());
  86. // distribution url must be read from /r/{packageName}/{version}.xml::/r/g:text()
  87. // but this location is 'de-facto' standard
  88. $distUrl = "http://{$packageDefinition->getChannelName()}/get/{$packageDefinition->getPackageName()}-{$version}.tgz";
  89. $requires = array();
  90. $suggests = array();
  91. $conflicts = array();
  92. $replaces = array();
  93. // alias package only when its channel matches repository channel,
  94. // cause we've know only repository channel alias
  95. if ($channelInfo->getName() == $packageDefinition->getChannelName()) {
  96. $composerPackageAlias = $this->buildComposerPackageName($channelInfo->getAlias(), $packageDefinition->getPackageName());
  97. $aliasConstraint = new VersionConstraint('==', $normalizedVersion);
  98. $replaces[] = new Link($composerPackageName, $composerPackageAlias, $aliasConstraint, 'replaces', (string) $aliasConstraint);
  99. }
  100. // alias package with user-specified prefix. it makes private pear channels looks like composer's.
  101. if (!empty($this->vendorAlias)) {
  102. $composerPackageAlias = "{$this->vendorAlias}/{$packageDefinition->getPackageName()}";
  103. $aliasConstraint = new VersionConstraint('==', $normalizedVersion);
  104. $replaces[] = new Link($composerPackageName, $composerPackageAlias, $aliasConstraint, 'replaces', (string) $aliasConstraint);
  105. }
  106. foreach ($releaseInfo->getDependencyInfo()->getRequires() as $dependencyConstraint) {
  107. $dependencyPackageName = $this->buildComposerPackageName($dependencyConstraint->getChannelName(), $dependencyConstraint->getPackageName());
  108. $constraint = $versionParser->parseConstraints($dependencyConstraint->getConstraint());
  109. $link = new Link($composerPackageName, $dependencyPackageName, $constraint, $dependencyConstraint->getType(), $dependencyConstraint->getConstraint());
  110. switch ($dependencyConstraint->getType()) {
  111. case 'required':
  112. $requires[] = $link;
  113. break;
  114. case 'conflicts':
  115. $conflicts[] = $link;
  116. break;
  117. case 'replaces':
  118. $replaces[] = $link;
  119. break;
  120. }
  121. }
  122. foreach ($releaseInfo->getDependencyInfo()->getOptionals() as $group => $dependencyConstraints) {
  123. foreach ($dependencyConstraints as $dependencyConstraint) {
  124. $dependencyPackageName = $this->buildComposerPackageName($dependencyConstraint->getChannelName(), $dependencyConstraint->getPackageName());
  125. $suggests[$group.'-'.$dependencyPackageName] = $dependencyConstraint->getConstraint();
  126. }
  127. }
  128. $package = new MemoryPackage($composerPackageName, $normalizedVersion, $version);
  129. $package->setType('library');
  130. $package->setDescription($packageDefinition->getDescription());
  131. $package->setDistType('pear');
  132. $package->setDistUrl($distUrl);
  133. $package->setAutoload(array('classmap' => array('')));
  134. $package->setIncludePaths(array('/'));
  135. $package->setRequires($requires);
  136. $package->setConflicts($conflicts);
  137. $package->setSuggests($suggests);
  138. $package->setReplaces($replaces);
  139. $result[] = $package;
  140. }
  141. }
  142. return $result;
  143. }
  144. private function buildComposerPackageName($channelName, $packageName)
  145. {
  146. if ('php' === $channelName) {
  147. return "php";
  148. }
  149. if ('ext' === $channelName) {
  150. return "ext-{$packageName}";
  151. }
  152. return "pear-{$channelName}/{$packageName}";
  153. }
  154. /**
  155. * Softened version parser.
  156. *
  157. * @param string $version
  158. * @return null|string
  159. */
  160. private function parseVersion($version)
  161. {
  162. if (preg_match('{^v?(\d{1,3})(\.\d+)?(\.\d+)?(\.\d+)?}i', $version, $matches)) {
  163. $version = $matches[1]
  164. .(!empty($matches[2]) ? $matches[2] : '.0')
  165. .(!empty($matches[3]) ? $matches[3] : '.0')
  166. .(!empty($matches[4]) ? $matches[4] : '.0');
  167. return $version;
  168. }
  169. return null;
  170. }
  171. }