VersionSelectorTest.php 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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 testHighestVersionIsReturned()
  72. {
  73. $packageName = 'foobar';
  74. $package1 = $this->createPackage('1.0.0');
  75. $package2 = $this->createPackage('1.1.0-beta');
  76. $packages = array($package1, $package2);
  77. $pool = $this->createMockPool();
  78. $pool->expects($this->once())
  79. ->method('whatProvides')
  80. ->with($packageName, null, true)
  81. ->will($this->returnValue($packages));
  82. $versionSelector = new VersionSelector($pool);
  83. $best = $versionSelector->findBestCandidate($packageName, null, null, 'dev');
  84. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  85. }
  86. public function testHighestVersionMatchingStabilityIsReturned()
  87. {
  88. $packageName = 'foobar';
  89. $package1 = $this->createPackage('1.0.0');
  90. $package2 = $this->createPackage('1.1.0-beta');
  91. $package3 = $this->createPackage('1.2.0-alpha');
  92. $packages = array($package1, $package2, $package3);
  93. $pool = $this->createMockPool();
  94. $pool->expects($this->once())
  95. ->method('whatProvides')
  96. ->with($packageName, null, true)
  97. ->will($this->returnValue($packages));
  98. $versionSelector = new VersionSelector($pool);
  99. $best = $versionSelector->findBestCandidate($packageName, null, null, 'beta');
  100. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  101. }
  102. public function testMostStableUnstableVersionIsReturned()
  103. {
  104. $packageName = 'foobar';
  105. $package2 = $this->createPackage('1.1.0-beta');
  106. $package3 = $this->createPackage('1.2.0-alpha');
  107. $packages = array($package2, $package3);
  108. $pool = $this->createMockPool();
  109. $pool->expects($this->once())
  110. ->method('whatProvides')
  111. ->with($packageName, null, true)
  112. ->will($this->returnValue($packages));
  113. $versionSelector = new VersionSelector($pool);
  114. $best = $versionSelector->findBestCandidate($packageName, null, null, 'stable');
  115. $this->assertSame($package2, $best, 'Latest version should be returned (1.1.0-beta)');
  116. }
  117. public function testFalseReturnedOnNoPackages()
  118. {
  119. $pool = $this->createMockPool();
  120. $pool->expects($this->once())
  121. ->method('whatProvides')
  122. ->will($this->returnValue(array()));
  123. $versionSelector = new VersionSelector($pool);
  124. $best = $versionSelector->findBestCandidate('foobaz');
  125. $this->assertFalse($best, 'No versions are available returns false');
  126. }
  127. /**
  128. * @dataProvider getRecommendedRequireVersionPackages
  129. */
  130. public function testFindRecommendedRequireVersion($prettyVersion, $isDev, $stability, $expectedVersion, $branchAlias = null)
  131. {
  132. $pool = $this->createMockPool();
  133. $versionSelector = new VersionSelector($pool);
  134. $versionParser = new VersionParser();
  135. $package = $this->getMock('\Composer\Package\PackageInterface');
  136. $package->expects($this->any())
  137. ->method('getPrettyVersion')
  138. ->will($this->returnValue($prettyVersion));
  139. $package->expects($this->any())
  140. ->method('getVersion')
  141. ->will($this->returnValue($versionParser->normalize($prettyVersion)));
  142. $package->expects($this->any())
  143. ->method('isDev')
  144. ->will($this->returnValue($isDev));
  145. $package->expects($this->any())
  146. ->method('getStability')
  147. ->will($this->returnValue($stability));
  148. $branchAlias = $branchAlias === null ? array() : array('branch-alias' => array($prettyVersion => $branchAlias));
  149. $package->expects($this->any())
  150. ->method('getExtra')
  151. ->will($this->returnValue($branchAlias));
  152. $recommended = $versionSelector->findRecommendedRequireVersion($package);
  153. // assert that the recommended version is what we expect
  154. $this->assertSame($expectedVersion, $recommended);
  155. }
  156. public function getRecommendedRequireVersionPackages()
  157. {
  158. return array(
  159. // real version, is dev package, stability, expected recommendation, [branch-alias]
  160. array('1.2.1', false, 'stable', '^1.2'),
  161. array('1.2', false, 'stable', '^1.2'),
  162. array('v1.2.1', false, 'stable', '^1.2'),
  163. array('3.1.2-pl2', false, 'stable', '^3.1'),
  164. array('3.1.2-patch', false, 'stable', '^3.1'),
  165. array('2.0-beta.1', false, 'beta', '^2.0@beta'),
  166. array('3.1.2-alpha5', false, 'alpha', '^3.1@alpha'),
  167. array('3.0-RC2', false, 'RC', '^3.0@RC'),
  168. array('0.1.0', false, 'stable', '^0.1.0'),
  169. array('0.1.3', false, 'stable', '^0.1.3'),
  170. array('0.0.3', false, 'stable', '^0.0.3'),
  171. array('0.0.3-alpha', false, 'alpha', '^0.0.3@alpha'),
  172. // date-based versions are not touched at all
  173. array('v20121020', false, 'stable', 'v20121020'),
  174. array('v20121020.2', false, 'stable', 'v20121020.2'),
  175. // dev packages without alias are not touched at all
  176. array('dev-master', true, 'dev', 'dev-master'),
  177. array('3.1.2-dev', true, 'dev', '3.1.2-dev'),
  178. // dev packages with alias inherit the alias
  179. array('dev-master', true, 'dev', '^2.1@dev', '2.1.x-dev'),
  180. array('dev-master', true, 'dev', '^2.1@dev', '2.1-dev'),
  181. array('dev-master', true, 'dev', '^2.1@dev', '2.1.3.x-dev'),
  182. array('dev-master', true, 'dev', '^2.0@dev', '2.x-dev'),
  183. array('dev-master', true, 'dev', '^0.3.0@dev', '0.3.x-dev'),
  184. array('dev-master', true, 'dev', '^0.0.3@dev', '0.0.3.x-dev'),
  185. // numeric alias
  186. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0.x-dev'),
  187. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0-dev'),
  188. );
  189. }
  190. private function createPackage($version)
  191. {
  192. $parser = new VersionParser();
  193. return new Package('foo', $parser->normalize($version), $version);
  194. }
  195. private function createMockPool()
  196. {
  197. return $this->getMock('Composer\DependencyResolver\Pool', array(), array(), '', true);
  198. }
  199. }