PearRepository.php 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  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\Package\Loader\ArrayLoader;
  13. /**
  14. * @author Benjamin Eberlei <kontakt@beberlei.de>
  15. * @author Jordi Boggiano <j.boggiano@seld.be>
  16. */
  17. class PearRepository extends ArrayRepository
  18. {
  19. protected $url;
  20. public function __construct(array $config)
  21. {
  22. if (!preg_match('{^https?://}', $config['url'])) {
  23. $config['url'] = 'http://'.$config['url'];
  24. }
  25. if (!filter_var($config['url'], FILTER_VALIDATE_URL)) {
  26. throw new \UnexpectedValueException('Invalid url given for PEAR repository: '.$config['url']);
  27. }
  28. $this->url = rtrim($config['url'], '/');
  29. }
  30. protected function initialize()
  31. {
  32. parent::initialize();
  33. set_error_handler(function($severity, $message, $file, $line) {
  34. throw new \ErrorException($message, $severity, $severity, $file, $line);
  35. });
  36. $this->fetchFromServer();
  37. restore_error_handler();
  38. }
  39. protected function fetchFromServer()
  40. {
  41. $categoryXML = $this->requestXml($this->url . "/rest/c/categories.xml");
  42. $categories = $categoryXML->getElementsByTagName("c");
  43. foreach ($categories as $category) {
  44. $link = $category->getAttribute("xlink:href");
  45. if (0 !== strpos($link, '/')) {
  46. $link = '/' . $link;
  47. }
  48. try {
  49. $packagesLink = str_replace("info.xml", "packagesinfo.xml", $link);
  50. $this->fetchPear2Packages($this->url . $packagesLink);
  51. } catch (\ErrorException $e) {
  52. if (false === strpos($e->getMessage(), '404')) {
  53. throw $e;
  54. }
  55. $categoryLink = str_replace("info.xml", "packages.xml", $link);
  56. $this->fetchPearPackages($this->url . $categoryLink);
  57. }
  58. }
  59. }
  60. /**
  61. * @param string $categoryLink
  62. * @throws ErrorException
  63. * @throws InvalidArgumentException
  64. */
  65. private function fetchPearPackages($categoryLink)
  66. {
  67. $packagesXML = $this->requestXml($categoryLink);
  68. $packages = $packagesXML->getElementsByTagName('p');
  69. $loader = new ArrayLoader();
  70. foreach ($packages as $package) {
  71. $packageName = $package->nodeValue;
  72. $packageLink = $package->getAttribute('xlink:href');
  73. $releaseLink = $this->url . str_replace("/rest/p/", "/rest/r/", $packageLink);
  74. $allReleasesLink = $releaseLink . "/allreleases2.xml";
  75. try {
  76. $releasesXML = $this->requestXml($allReleasesLink);
  77. } catch (\ErrorException $e) {
  78. if (strpos($e->getMessage(), '404')) {
  79. continue;
  80. }
  81. throw $e;
  82. }
  83. $releases = $releasesXML->getElementsByTagName('r');
  84. foreach ($releases as $release) {
  85. /* @var $release \DOMElement */
  86. $pearVersion = $release->getElementsByTagName('v')->item(0)->nodeValue;
  87. $packageData = array(
  88. 'name' => $packageName,
  89. 'type' => 'library',
  90. 'dist' => array('type' => 'pear', 'url' => $this->url.'/get/'.$packageName.'-'.$pearVersion.".tgz"),
  91. 'version' => $pearVersion,
  92. );
  93. try {
  94. $deps = file_get_contents($releaseLink . "/deps.".$pearVersion.".txt");
  95. } catch (\ErrorException $e) {
  96. if (strpos($e->getMessage(), '404')) {
  97. continue;
  98. }
  99. throw $e;
  100. }
  101. $packageData += $this->parseDependencies($deps);
  102. try {
  103. $this->addPackage($loader->load($packageData));
  104. } catch (\UnexpectedValueException $e) {
  105. continue;
  106. }
  107. }
  108. }
  109. }
  110. /**
  111. * @param array $data
  112. * @return string
  113. */
  114. private function parseVersion(array $data)
  115. {
  116. if (!isset($data['min']) && !isset($data['max'])) {
  117. return '*';
  118. }
  119. $versions = array();
  120. if (isset($data['min'])) {
  121. $versions[] = '>=' . $data['min'];
  122. }
  123. if (isset($data['max'])) {
  124. $versions[] = '<=' . $data['max'];
  125. }
  126. return implode(',', $versions);
  127. }
  128. /**
  129. * @todo Improve dependencies resolution of pear packages.
  130. * @param array $options
  131. * @return array
  132. */
  133. private function parseDependenciesOptions(array $depsOptions)
  134. {
  135. $data = array();
  136. foreach ($depsOptions as $name => $options) {
  137. // make sure single deps are wrapped in an array
  138. if (isset($options['name'])) {
  139. $options = array($options);
  140. }
  141. if ('php' == $name) {
  142. $data[$name] = $this->parseVersion($options);
  143. } elseif ('package' == $name) {
  144. foreach ($options as $key => $value) {
  145. if (is_array($value)) {
  146. $dataKey = $value['name'];
  147. $data[$dataKey] = $this->parseVersion($value);
  148. }
  149. }
  150. } elseif ('extension' == $name) {
  151. foreach ($options as $key => $value) {
  152. $dataKey = 'ext-' . $value['name'];
  153. $data[$dataKey] = $this->parseVersion($value);
  154. }
  155. }
  156. }
  157. return $data;
  158. }
  159. /**
  160. * @param string $deps
  161. * @return array
  162. * @throws InvalidArgumentException
  163. */
  164. private function parseDependencies($deps)
  165. {
  166. if (preg_match('((O:([0-9])+:"([^"]+)"))', $deps, $matches)) {
  167. if (strlen($matches[3]) == $matches[2]) {
  168. throw new \InvalidArgumentException("Invalid dependency data, it contains serialized objects.");
  169. }
  170. }
  171. $deps = (array) @unserialize($deps);
  172. unset($deps['required']['pearinstaller']);
  173. $depsData = array();
  174. if (!empty($deps['required'])) {
  175. $depsData['require'] = $this->parseDependenciesOptions($deps['required']);
  176. }
  177. if (!empty($deps['optional'])) {
  178. $depsData['suggest'] = $this->parseDependenciesOptions($deps['optional']);
  179. }
  180. return $depsData;
  181. }
  182. /**
  183. * @param string $packagesLink
  184. * @return void
  185. * @throws InvalidArgumentException
  186. */
  187. private function fetchPear2Packages($packagesLink)
  188. {
  189. $loader = new ArrayLoader();
  190. $packagesXml = $this->requestXml($packagesLink);
  191. $informations = $packagesXml->getElementsByTagName('pi');
  192. foreach ($informations as $information) {
  193. $package = $information->getElementsByTagName('p')->item(0);
  194. $packageName = $package->getElementsByTagName('n')->item(0)->nodeValue;
  195. $packageData = array(
  196. 'name' => $packageName,
  197. 'type' => 'library'
  198. );
  199. $packageKeys = array('l' => 'license', 'd' => 'description');
  200. foreach ($packageKeys as $pear => $composer) {
  201. if ($package->getElementsByTagName($pear)->length > 0
  202. && ($pear = $package->getElementsByTagName($pear)->item(0)->nodeValue)) {
  203. $packageData[$composer] = $pear;
  204. }
  205. }
  206. $depsData = array();
  207. foreach ($information->getElementsByTagName('deps') as $depElement) {
  208. $depsVersion = $depElement->getElementsByTagName('v')->item(0)->nodeValue;
  209. $depsData[$depsVersion] = $this->parseDependencies(
  210. $depElement->getElementsByTagName('d')->item(0)->nodeValue
  211. );
  212. }
  213. $releases = $information->getElementsByTagName('a')->item(0);
  214. if (!$releases) {
  215. continue;
  216. }
  217. $releases = $releases->getElementsByTagName('r');
  218. $packageUrl = $this->url . '/get/' . $packageName;
  219. foreach ($releases as $release) {
  220. $version = $release->getElementsByTagName('v')->item(0)->nodeValue;
  221. $releaseData = array(
  222. 'dist' => array(
  223. 'type' => 'pear',
  224. 'url' => $packageUrl . '-' . $version . '.tgz'
  225. ),
  226. 'version' => $version
  227. );
  228. if (isset($depsData[$version])) {
  229. $releaseData += $depsData[$version];
  230. }
  231. try {
  232. $this->addPackage(
  233. $loader->load($packageData + $releaseData)
  234. );
  235. } catch (\UnexpectedValueException $e) {
  236. continue;
  237. }
  238. }
  239. }
  240. }
  241. /**
  242. * @param string $url
  243. * @return DOMDocument
  244. */
  245. private function requestXml($url)
  246. {
  247. $content = file_get_contents($url);
  248. if (!$content) {
  249. throw new \UnexpectedValueException('The PEAR channel at '.$url.' did not respond.');
  250. }
  251. $dom = new \DOMDocument('1.0', 'UTF-8');
  252. $dom->loadXML($content);
  253. return $dom;
  254. }
  255. }