VersionSelectorTest.php 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  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 testFalseReturnedOnNoPackages()
  103. {
  104. $pool = $this->createMockPool();
  105. $pool->expects($this->once())
  106. ->method('whatProvides')
  107. ->will($this->returnValue(array()));
  108. $versionSelector = new VersionSelector($pool);
  109. $best = $versionSelector->findBestCandidate('foobaz');
  110. $this->assertFalse($best, 'No versions are available returns false');
  111. }
  112. /**
  113. * @dataProvider getRecommendedRequireVersionPackages
  114. */
  115. public function testFindRecommendedRequireVersion($prettyVersion, $isDev, $stability, $expectedVersion, $branchAlias = null)
  116. {
  117. $pool = $this->createMockPool();
  118. $versionSelector = new VersionSelector($pool);
  119. $versionParser = new VersionParser();
  120. $package = $this->getMock('\Composer\Package\PackageInterface');
  121. $package->expects($this->any())
  122. ->method('getPrettyVersion')
  123. ->will($this->returnValue($prettyVersion));
  124. $package->expects($this->any())
  125. ->method('getVersion')
  126. ->will($this->returnValue($versionParser->normalize($prettyVersion)));
  127. $package->expects($this->any())
  128. ->method('isDev')
  129. ->will($this->returnValue($isDev));
  130. $package->expects($this->any())
  131. ->method('getStability')
  132. ->will($this->returnValue($stability));
  133. $branchAlias = $branchAlias === null ? array() : array('branch-alias' => array($prettyVersion => $branchAlias));
  134. $package->expects($this->any())
  135. ->method('getExtra')
  136. ->will($this->returnValue($branchAlias));
  137. $recommended = $versionSelector->findRecommendedRequireVersion($package);
  138. // assert that the recommended version is what we expect
  139. $this->assertSame($expectedVersion, $recommended);
  140. }
  141. public function getRecommendedRequireVersionPackages()
  142. {
  143. return array(
  144. // real version, is dev package, stability, expected recommendation, [branch-alias]
  145. array('1.2.1', false, 'stable', '^1.2'),
  146. array('1.2', false, 'stable', '^1.2'),
  147. array('v1.2.1', false, 'stable', '^1.2'),
  148. array('3.1.2-pl2', false, 'stable', '^3.1'),
  149. array('3.1.2-patch', false, 'stable', '^3.1'),
  150. array('2.0-beta.1', false, 'beta', '^2.0@beta'),
  151. array('3.1.2-alpha5', false, 'alpha', '^3.1@alpha'),
  152. array('3.0-RC2', false, 'RC', '^3.0@RC'),
  153. array('0.1.0', false, 'stable', '^0.1.0'),
  154. array('0.1.3', false, 'stable', '^0.1.3'),
  155. array('0.0.3', false, 'stable', '^0.0.3'),
  156. array('0.0.3-alpha', false, 'alpha', '^0.0.3@alpha'),
  157. // date-based versions are not touched at all
  158. array('v20121020', false, 'stable', 'v20121020'),
  159. array('v20121020.2', false, 'stable', 'v20121020.2'),
  160. // dev packages without alias are not touched at all
  161. array('dev-master', true, 'dev', 'dev-master'),
  162. array('3.1.2-dev', true, 'dev', '3.1.2-dev'),
  163. // dev packages with alias inherit the alias
  164. array('dev-master', true, 'dev', '^2.1@dev', '2.1.x-dev'),
  165. array('dev-master', true, 'dev', '^2.1@dev', '2.1-dev'),
  166. array('dev-master', true, 'dev', '^2.1@dev', '2.1.3.x-dev'),
  167. array('dev-master', true, 'dev', '^2.0@dev', '2.x-dev'),
  168. array('dev-master', true, 'dev', '^0.3.0@dev', '0.3.x-dev'),
  169. array('dev-master', true, 'dev', '^0.0.3@dev', '0.0.3.x-dev'),
  170. // numeric alias
  171. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0.x-dev'),
  172. array('3.x-dev', true, 'dev', '^3.0@dev', '3.0-dev'),
  173. );
  174. }
  175. private function createPackage($version)
  176. {
  177. $parser = new VersionParser();
  178. return new Package('foo', $parser->normalize($version), $version);
  179. }
  180. private function createMockPool()
  181. {
  182. return $this->getMock('Composer\DependencyResolver\Pool', array(), array(), '', true);
  183. }
  184. }