VersionSelectorTest.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  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\Test\Package\Version;
  12. use Composer\Package\Version\VersionSelector;
  13. use Composer\Package\Package;
  14. use Composer\Package\Link;
  15. use Composer\Semver\VersionParser;
  16. class VersionSelectorTest extends \PHPUnit_Framework_TestCase
  17. {
  18. // A) multiple versions, get the latest one
  19. // B) targetPackageVersion will pass to pool
  20. // C) No results, throw exception
  21. public function testLatestVersionIsReturned()
  22. {
  23. $packageName = 'foobar';
  24. $package1 = $this->createPackage('1.2.1');
  25. $package2 = $this->createPackage('1.2.2');
  26. $package3 = $this->createPackage('1.2.0');
  27. $packages = array($package1, $package2, $package3);
  28. $pool = $this->createMockPool();
  29. $pool->expects($this->once())
  30. ->method('whatProvides')
  31. ->with($packageName, null, true)
  32. ->will($this->returnValue($packages));
  33. $versionSelector = new VersionSelector($pool);
  34. $best = $versionSelector->findBestCandidate($packageName);
  35. // 1.2.2 should be returned because it's the latest of the returned versions
  36. $this->assertSame($package2, $best, 'Latest version should be 1.2.2');
  37. }
  38. public function testLatestVersionIsReturnedThatMatchesPhpRequirement()
  39. {
  40. $packageName = 'foobar';
  41. $parser = new VersionParser;
  42. $package1 = $this->createPackage('1.0.0');
  43. $package2 = $this->createPackage('2.0.0');
  44. $package1->setRequires(array('php' => new Link($packageName, 'php', $parser->parseConstraints('>=5.4'), 'requires', '>=5.4')));
  45. $package2->setRequires(array('php' => new Link($packageName, 'php', $parser->parseConstraints('>=5.6'), 'requires', '>=5.6')));
  46. $packages = array($package1, $package2);
  47. $pool = $this->createMockPool();
  48. $pool->expects($this->once())
  49. ->method('whatProvides')
  50. ->with($packageName, null, true)
  51. ->will($this->returnValue($packages));
  52. $versionSelector = new VersionSelector($pool);
  53. $best = $versionSelector->findBestCandidate($packageName, null, '5.5.0');
  54. $this->assertSame($package1, $best, 'Latest version supporting php 5.5 should be returned (1.0.0)');
  55. }
  56. public function testMostStableVersionIsReturned()
  57. {
  58. $packageName = 'foobar';
  59. $package1 = $this->createPackage('1.0.0');
  60. $package2 = $this->createPackage('1.1.0-beta');
  61. $packages = array($package1, $package2);
  62. $pool = $this->createMockPool();
  63. $pool->expects($this->once())
  64. ->method('whatProvides')
  65. ->with($packageName, null, true)
  66. ->will($this->returnValue($packages));
  67. $versionSelector = new VersionSelector($pool);
  68. $best = $versionSelector->findBestCandidate($packageName);
  69. $this->assertSame($package1, $best, 'Latest most stable version should be returned (1.0.0)');
  70. }
  71. public function testMostStableVersionIsReturnedRegardlessOfOrder()
  72. {
  73. $packageName = 'foobar';
  74. $package1 = $this->createPackage('2.x-dev');
  75. $package2 = $this->createPackage('2.0.0-beta3');
  76. $packages = array($package1, $package2);
  77. $pool = $this->createMockPool();
  78. $pool->expects($this->at(0))
  79. ->method('whatProvides')
  80. ->with($packageName, null, true)
  81. ->will($this->returnValue($packages));
  82. $pool->expects($this->at(1))
  83. ->method('whatProvides')
  84. ->with($packageName, null, true)
  85. ->will($this->returnValue(array_reverse($packages)));
  86. $versionSelector = new VersionSelector($pool);
  87. $best = $versionSelector->findBestCandidate($packageName, null, null);
  88. $this->assertSame($package2, $best, 'Expecting 2.0.0-beta3, cause beta is more stable than dev');
  89. $best = $versionSelector->findBestCandidate($packageName, null, null);
  90. $this->assertSame($package2, $best, 'Expecting 2.0.0-beta3, cause beta is more stable than dev');
  91. }
  92. public function testHighestVersionIsReturned()
  93. {
  94. $packageName = 'foobar';
  95. $package1 = $this->createPackage('1.0.0');
  96. $package2 = $this->createPackage('1.1.0-beta');
  97. $packages = array($package1, $package2);
  98. $pool = $this->createMockPool();
  99. $pool->expects($this->once())
  100. ->method('whatProvides')
  101. ->with($packageName, null, true)
  102. ->will($this->returnValue($packages));
  103. $versionSelector = new VersionSelector($pool);
  104. $best = $versionSelector->findBestCandidate($packageName, null, null, 'dev');
  105. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  106. }
  107. public function testHighestVersionMatchingStabilityIsReturned()
  108. {
  109. $packageName = 'foobar';
  110. $package1 = $this->createPackage('1.0.0');
  111. $package2 = $this->createPackage('1.1.0-beta');
  112. $package3 = $this->createPackage('1.2.0-alpha');
  113. $packages = array($package1, $package2, $package3);
  114. $pool = $this->createMockPool();
  115. $pool->expects($this->once())
  116. ->method('whatProvides')
  117. ->with($packageName, null, true)
  118. ->will($this->returnValue($packages));
  119. $versionSelector = new VersionSelector($pool);
  120. $best = $versionSelector->findBestCandidate($packageName, null, null, 'beta');
  121. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  122. }
  123. public function testMostStableUnstableVersionIsReturned()
  124. {
  125. $packageName = 'foobar';
  126. $package2 = $this->createPackage('1.1.0-beta');
  127. $package3 = $this->createPackage('1.2.0-alpha');
  128. $packages = array($package2, $package3);
  129. $pool = $this->createMockPool();
  130. $pool->expects($this->once())
  131. ->method('whatProvides')
  132. ->with($packageName, null, true)
  133. ->will($this->returnValue($packages));
  134. $versionSelector = new VersionSelector($pool);
  135. $best = $versionSelector->findBestCandidate($packageName, null, null, 'stable');
  136. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  137. }
  138. public function testFalseReturnedOnNoPackages()
  139. {
  140. $pool = $this->createMockPool();
  141. $pool->expects($this->once())
  142. ->method('whatProvides')
  143. ->will($this->returnValue(array()));
  144. $versionSelector = new VersionSelector($pool);
  145. $best = $versionSelector->findBestCandidate('foobaz');
  146. $this->assertFalse($best, 'No versions are available returns false');
  147. }
  148. /**
  149. * @dataProvider getRecommendedRequireVersionPackages
  150. */
  151. public function testFindRecommendedRequireVersion($prettyVersion, $isDev, $stability, $expectedVersion, $branchAlias = null)
  152. {
  153. $pool = $this->createMockPool();
  154. $versionSelector = new VersionSelector($pool);
  155. $versionParser = new VersionParser();
  156. $package = $this->getMock('\Composer\Package\PackageInterface');
  157. $package->expects($this->any())
  158. ->method('getPrettyVersion')
  159. ->will($this->returnValue($prettyVersion));
  160. $package->expects($this->any())
  161. ->method('getVersion')
  162. ->will($this->returnValue($versionParser->normalize($prettyVersion)));
  163. $package->expects($this->any())
  164. ->method('isDev')
  165. ->will($this->returnValue($isDev));
  166. $package->expects($this->any())
  167. ->method('getStability')
  168. ->will($this->returnValue($stability));
  169. $branchAlias = $branchAlias === null ? array() : array('branch-alias' => array($prettyVersion => $branchAlias));
  170. $package->expects($this->any())
  171. ->method('getExtra')
  172. ->will($this->returnValue($branchAlias));
  173. $recommended = $versionSelector->findRecommendedRequireVersion($package);
  174. // assert that the recommended version is what we expect
  175. $this->assertSame($expectedVersion, $recommended);
  176. }
  177. public function getRecommendedRequireVersionPackages()
  178. {
  179. return array(
  180. // real version, is dev package, stability, expected recommendation, [branch-alias]
  181. array('1.2.1', false, 'stable', '^1.2'),
  182. array('1.2', false, 'stable', '^1.2'),
  183. array('v1.2.1', false, 'stable', '^1.2'),
  184. array('3.1.2-pl2', false, 'stable', '^3.1'),
  185. array('3.1.2-patch', false, 'stable', '^3.1'),
  186. array('2.0-beta.1', false, 'beta', '^2.0@beta'),
  187. array('3.1.2-alpha5', false, 'alpha', '^3.1@alpha'),
  188. array('3.0-RC2', false, 'RC', '^3.0@RC'),
  189. array('0.1.0', false, 'stable', '^0.1.0'),
  190. array('0.1.3', false, 'stable', '^0.1.3'),
  191. array('0.0.3', false, 'stable', '^0.0.3'),
  192. array('0.0.3-alpha', false, 'alpha', '^0.0.3@alpha'),
  193. // date-based versions are not touched at all
  194. array('v20121020', false, 'stable', 'v20121020'),
  195. array('v20121020.2', false, 'stable', 'v20121020.2'),
  196. // dev packages without alias are not touched at all
  197. array('dev-master', true, 'dev', 'dev-master'),
  198. array('3.1.2-dev', true, 'dev', '3.1.2-dev'),
  199. // dev packages with alias inherit the alias
  200. array('dev-master', true, 'dev', '^2.1@dev', '2.1.x-dev'),
  201. array('dev-master', true, 'dev', '^2.1@dev', '2.1-dev'),
  202. array('dev-master', true, 'dev', '^2.1@dev', '2.1.3.x-dev'),
  203. array('dev-master', true, 'dev', '^2.0@dev', '2.x-dev'),
  204. array('dev-master', true, 'dev', '^0.3.0@dev', '0.3.x-dev'),
  205. array('dev-master', true, 'dev', '^0.0.3@dev', '0.0.3.x-dev'),
  206. // numeric alias
  207. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0.x-dev'),
  208. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0-dev'),
  209. );
  210. }
  211. private function createPackage($version)
  212. {
  213. $parser = new VersionParser();
  214. return new Package('foo', $parser->normalize($version), $version);
  215. }
  216. private function createMockPool()
  217. {
  218. return $this->getMock('Composer\DependencyResolver\Pool', array(), array(), '', true);
  219. }
  220. }