PearRepository.php 13 KB

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