PearRepository.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  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\Loader\ArrayLoader;
  14. use Composer\Util\RemoteFilesystem;
  15. use Composer\Json\JsonFile;
  16. use Composer\Config;
  17. use Composer\Downloader\TransportException;
  18. /**
  19. * @author Benjamin Eberlei <kontakt@beberlei.de>
  20. * @author Jordi Boggiano <j.boggiano@seld.be>
  21. */
  22. class PearRepository extends ArrayRepository
  23. {
  24. private static $channelNames = array();
  25. private $url;
  26. private $channel;
  27. private $io;
  28. private $rfs;
  29. public function __construct(array $repoConfig, IOInterface $io, Config $config, RemoteFilesystem $rfs = null)
  30. {
  31. if (!preg_match('{^https?://}', $repoConfig['url'])) {
  32. $repoConfig['url'] = 'http://'.$repoConfig['url'];
  33. }
  34. if (function_exists('filter_var') && !filter_var($repoConfig['url'], FILTER_VALIDATE_URL)) {
  35. throw new \UnexpectedValueException('Invalid url given for PEAR repository: '.$repoConfig['url']);
  36. }
  37. $this->url = rtrim($repoConfig['url'], '/');
  38. $this->channel = !empty($repoConfig['channel']) ? $repoConfig['channel'] : null;
  39. $this->io = $io;
  40. $this->rfs = $rfs ?: new RemoteFilesystem($this->io);
  41. }
  42. protected function initialize()
  43. {
  44. parent::initialize();
  45. $this->io->write('Initializing PEAR repository '.$this->url);
  46. $this->initializeChannel();
  47. $this->io->write('Packages names will be prefixed with: pear-'.$this->channel.'/');
  48. // try to load as a composer repo
  49. try {
  50. $json = new JsonFile($this->url.'/packages.json', new RemoteFilesystem($this->io));
  51. $packages = $json->read();
  52. if ($this->io->isVerbose()) {
  53. $this->io->write('Repository is Composer-compatible, loading via packages.json instead of PEAR protocol');
  54. }
  55. $loader = new ArrayLoader();
  56. foreach ($packages as $data) {
  57. foreach ($data['versions'] as $rev) {
  58. $rev['name'] = 'pear-'.$this->channel.'/'.$rev['name'];
  59. $this->addPackage($loader->load($rev));
  60. }
  61. }
  62. return;
  63. } catch (\Exception $e) {
  64. }
  65. $this->fetchFromServer();
  66. }
  67. protected function initializeChannel()
  68. {
  69. $channelXML = $this->requestXml($this->url . "/channel.xml");
  70. if (!$this->channel) {
  71. $this->channel = $channelXML->getElementsByTagName("suggestedalias")->item(0)->nodeValue
  72. ?: $channelXML->getElementsByTagName("name")->item(0)->nodeValue;
  73. }
  74. self::$channelNames[$channelXML->getElementsByTagName("name")->item(0)->nodeValue] = $this->channel;
  75. }
  76. protected function fetchFromServer()
  77. {
  78. $categoryXML = $this->requestXml($this->url . "/rest/c/categories.xml");
  79. $categories = $categoryXML->getElementsByTagName("c");
  80. foreach ($categories as $category) {
  81. $link = '/' . ltrim($category->getAttribute("xlink:href"), '/');
  82. try {
  83. $packagesLink = str_replace("info.xml", "packagesinfo.xml", $link);
  84. $this->fetchPear2Packages($this->url . $packagesLink);
  85. } catch (TransportException $e) {
  86. if (false === strpos($e->getMessage(), '404')) {
  87. throw $e;
  88. }
  89. $categoryLink = str_replace("info.xml", "packages.xml", $link);
  90. $this->fetchPearPackages($this->url . $categoryLink);
  91. }
  92. }
  93. }
  94. /**
  95. * @param string $categoryLink
  96. * @throws TransportException
  97. * @throws InvalidArgumentException
  98. */
  99. private function fetchPearPackages($categoryLink)
  100. {
  101. $packagesXML = $this->requestXml($categoryLink);
  102. $packages = $packagesXML->getElementsByTagName('p');
  103. $loader = new ArrayLoader();
  104. foreach ($packages as $package) {
  105. $packageName = $package->nodeValue;
  106. $fullName = 'pear-'.$this->channel.'/'.$packageName;
  107. $packageLink = $package->getAttribute('xlink:href');
  108. $releaseLink = $this->url . str_replace("/rest/p/", "/rest/r/", $packageLink);
  109. $allReleasesLink = $releaseLink . "/allreleases2.xml";
  110. try {
  111. $releasesXML = $this->requestXml($allReleasesLink);
  112. } catch (TransportException $e) {
  113. if (strpos($e->getMessage(), '404')) {
  114. continue;
  115. }
  116. throw $e;
  117. }
  118. $releases = $releasesXML->getElementsByTagName('r');
  119. foreach ($releases as $release) {
  120. /* @var $release \DOMElement */
  121. $pearVersion = $release->getElementsByTagName('v')->item(0)->nodeValue;
  122. $packageData = array(
  123. 'name' => $fullName,
  124. 'type' => 'library',
  125. 'dist' => array('type' => 'pear', 'url' => $this->url.'/get/'.$packageName.'-'.$pearVersion.".tgz"),
  126. 'version' => $pearVersion,
  127. 'autoload' => array(
  128. 'classmap' => array(''),
  129. ),
  130. );
  131. try {
  132. $deps = $this->rfs->getContents($this->url, $releaseLink . "/deps.".$pearVersion.".txt", false);
  133. } catch (TransportException $e) {
  134. if (strpos($e->getMessage(), '404')) {
  135. continue;
  136. }
  137. throw $e;
  138. }
  139. $packageData += $this->parseDependencies($deps);
  140. try {
  141. $this->addPackage($loader->load($packageData));
  142. if ($this->io->isVerbose()) {
  143. $this->io->write('Loaded '.$packageData['name'].' '.$packageData['version']);
  144. }
  145. } catch (\UnexpectedValueException $e) {
  146. if ($this->io->isVerbose()) {
  147. $this->io->write('Could not load '.$packageData['name'].' '.$packageData['version'].': '.$e->getMessage());
  148. }
  149. continue;
  150. }
  151. }
  152. }
  153. }
  154. /**
  155. * @param array $data
  156. * @return string
  157. */
  158. private function parseVersion(array $data)
  159. {
  160. if (!isset($data['min']) && !isset($data['max'])) {
  161. return '*';
  162. }
  163. $versions = array();
  164. if (isset($data['min'])) {
  165. $versions[] = '>=' . $data['min'];
  166. }
  167. if (isset($data['max'])) {
  168. $versions[] = '<=' . $data['max'];
  169. }
  170. return implode(',', $versions);
  171. }
  172. /**
  173. * @todo Improve dependencies resolution of pear packages.
  174. * @param array $options
  175. * @return array
  176. */
  177. private function parseDependenciesOptions(array $depsOptions)
  178. {
  179. $data = array();
  180. foreach ($depsOptions as $name => $options) {
  181. // make sure single deps are wrapped in an array
  182. if (isset($options['name'])) {
  183. $options = array($options);
  184. }
  185. if ('php' == $name) {
  186. $data[$name] = $this->parseVersion($options);
  187. } elseif ('package' == $name) {
  188. foreach ($options as $key => $value) {
  189. if (isset($value['providesextension'])) {
  190. // skip PECL dependencies
  191. continue;
  192. }
  193. if (isset($value['uri'])) {
  194. // skip uri-based dependencies
  195. continue;
  196. }
  197. if (is_array($value)) {
  198. $dataKey = $value['name'];
  199. if (false === strpos($dataKey, '/')) {
  200. $dataKey = $this->getChannelShorthand($value['channel']).'/'.$dataKey;
  201. }
  202. $data['pear-'.$dataKey] = $this->parseVersion($value);
  203. }
  204. }
  205. } elseif ('extension' == $name) {
  206. foreach ($options as $key => $value) {
  207. $dataKey = 'ext-' . $value['name'];
  208. $data[$dataKey] = $this->parseVersion($value);
  209. }
  210. }
  211. }
  212. return $data;
  213. }
  214. /**
  215. * @param string $deps
  216. * @return array
  217. * @throws InvalidArgumentException
  218. */
  219. private function parseDependencies($deps)
  220. {
  221. if (preg_match('((O:([0-9])+:"([^"]+)"))', $deps, $matches)) {
  222. if (strlen($matches[3]) == $matches[2]) {
  223. throw new \InvalidArgumentException("Invalid dependency data, it contains serialized objects.");
  224. }
  225. }
  226. $deps = (array) @unserialize($deps);
  227. unset($deps['required']['pearinstaller']);
  228. $depsData = array();
  229. if (!empty($deps['required'])) {
  230. $depsData['require'] = $this->parseDependenciesOptions($deps['required']);
  231. }
  232. if (!empty($deps['optional'])) {
  233. $depsData['suggest'] = $this->parseDependenciesOptions($deps['optional']);
  234. }
  235. return $depsData;
  236. }
  237. /**
  238. * @param string $packagesLink
  239. * @return void
  240. * @throws InvalidArgumentException
  241. */
  242. private function fetchPear2Packages($packagesLink)
  243. {
  244. $loader = new ArrayLoader();
  245. $packagesXml = $this->requestXml($packagesLink);
  246. $informations = $packagesXml->getElementsByTagName('pi');
  247. foreach ($informations as $information) {
  248. $package = $information->getElementsByTagName('p')->item(0);
  249. $packageName = $package->getElementsByTagName('n')->item(0)->nodeValue;
  250. $fullName = 'pear-'.$this->channel.'/'.$packageName;
  251. $packageData = array(
  252. 'name' => $fullName,
  253. 'type' => 'library',
  254. 'autoload' => array(
  255. 'classmap' => array(''),
  256. ),
  257. );
  258. $packageKeys = array('l' => 'license', 'd' => 'description');
  259. foreach ($packageKeys as $pear => $composer) {
  260. if ($package->getElementsByTagName($pear)->length > 0
  261. && ($pear = $package->getElementsByTagName($pear)->item(0)->nodeValue)) {
  262. $packageData[$composer] = $pear;
  263. }
  264. }
  265. $depsData = array();
  266. foreach ($information->getElementsByTagName('deps') as $depElement) {
  267. $depsVersion = $depElement->getElementsByTagName('v')->item(0)->nodeValue;
  268. $depsData[$depsVersion] = $this->parseDependencies(
  269. $depElement->getElementsByTagName('d')->item(0)->nodeValue
  270. );
  271. }
  272. $releases = $information->getElementsByTagName('a')->item(0);
  273. if (!$releases) {
  274. continue;
  275. }
  276. $releases = $releases->getElementsByTagName('r');
  277. $packageUrl = $this->url . '/get/' . $packageName;
  278. foreach ($releases as $release) {
  279. $version = $release->getElementsByTagName('v')->item(0)->nodeValue;
  280. $releaseData = array(
  281. 'dist' => array(
  282. 'type' => 'pear',
  283. 'url' => $packageUrl . '-' . $version . '.tgz'
  284. ),
  285. 'version' => $version
  286. );
  287. if (isset($depsData[$version])) {
  288. $releaseData += $depsData[$version];
  289. }
  290. $package = $packageData + $releaseData;
  291. try {
  292. $this->addPackage($loader->load($package));
  293. if ($this->io->isVerbose()) {
  294. $this->io->write('Loaded '.$package['name'].' '.$package['version']);
  295. }
  296. } catch (\UnexpectedValueException $e) {
  297. if ($this->io->isVerbose()) {
  298. $this->io->write('Could not load '.$package['name'].' '.$package['version'].': '.$e->getMessage());
  299. }
  300. continue;
  301. }
  302. }
  303. }
  304. }
  305. /**
  306. * @param string $url
  307. * @return DOMDocument
  308. */
  309. private function requestXml($url)
  310. {
  311. $content = $this->rfs->getContents($this->url, $url, false);
  312. if (!$content) {
  313. throw new \UnexpectedValueException('The PEAR channel at '.$url.' did not respond.');
  314. }
  315. $dom = new \DOMDocument('1.0', 'UTF-8');
  316. $dom->loadXML($content);
  317. return $dom;
  318. }
  319. private function getChannelShorthand($url)
  320. {
  321. if (!isset(self::$channelNames[$url])) {
  322. try {
  323. $channelXML = $this->requestXml('http://'.$url."/channel.xml");
  324. $shorthand = $channelXML->getElementsByTagName("suggestedalias")->item(0)->nodeValue
  325. ?: $channelXML->getElementsByTagName("name")->item(0)->nodeValue;
  326. self::$channelNames[$url] = $shorthand;
  327. } catch (\Exception $e) {
  328. self::$channelNames[$url] = substr($url, 0, strpos($url, '.'));
  329. }
  330. }
  331. return self::$channelNames[$url];
  332. }
  333. }