VersionParserTest.php 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870
  1. <?php
  2. /*
  3. * This file is part of composer/semver.
  4. *
  5. * (c) Composer <https://github.com/composer>
  6. *
  7. * For the full copyright and license information, please view
  8. * the LICENSE file that was distributed with this source code.
  9. */
  10. namespace Composer\Semver;
  11. use Composer\Semver\Constraint\MatchAllConstraint;
  12. use Composer\Semver\Constraint\MultiConstraint;
  13. use Composer\Semver\Constraint\Constraint;
  14. use PHPUnit\Framework\TestCase;
  15. class VersionParserTest extends TestCase
  16. {
  17. /**
  18. * @dataProvider numericAliasVersions
  19. * @param string $input
  20. * @param string $expected
  21. */
  22. public function testParseNumericAliasPrefix($input, $expected)
  23. {
  24. $parser = new VersionParser();
  25. $this->assertSame($expected, $parser->parseNumericAliasPrefix($input));
  26. }
  27. /**
  28. * @return array<mixed>
  29. */
  30. public function numericAliasVersions()
  31. {
  32. return array(
  33. array('0.x-dev', '0.'),
  34. array('1.0.x-dev', '1.0.'),
  35. array('1.x-dev', '1.'),
  36. array('1.2.x-dev', '1.2.'),
  37. array('1.2-dev', '1.2.'),
  38. array('1-dev', '1.'),
  39. array('dev-develop', false),
  40. array('dev-master', false),
  41. );
  42. }
  43. /**
  44. * @dataProvider successfulNormalizedVersions
  45. * @param string $input
  46. * @param string $expected
  47. */
  48. public function testNormalizeSucceeds($input, $expected)
  49. {
  50. $parser = new VersionParser();
  51. $this->assertSame($expected, $parser->normalize($input));
  52. }
  53. /**
  54. * @return array<mixed>
  55. */
  56. public function successfulNormalizedVersions()
  57. {
  58. return array(
  59. 'none' => array('1.0.0', '1.0.0.0'),
  60. 'none/2' => array('1.2.3.4', '1.2.3.4'),
  61. 'parses state' => array('1.0.0RC1dev', '1.0.0.0-RC1-dev'),
  62. 'CI parsing' => array('1.0.0-rC15-dev', '1.0.0.0-RC15-dev'),
  63. 'delimiters' => array('1.0.0.RC.15-dev', '1.0.0.0-RC15-dev'),
  64. 'RC uppercase' => array('1.0.0-rc1', '1.0.0.0-RC1'),
  65. 'patch replace' => array('1.0.0.pl3-dev', '1.0.0.0-patch3-dev'),
  66. 'forces w.x.y.z' => array('1.0-dev', '1.0.0.0-dev'),
  67. 'forces w.x.y.z/2' => array('0', '0.0.0.0'),
  68. 'forces w.x.y.z/maximum major' => array('99999', '99999.0.0.0'),
  69. 'parses long' => array('10.4.13-beta', '10.4.13.0-beta'),
  70. 'parses long/2' => array('10.4.13beta2', '10.4.13.0-beta2'),
  71. 'parses long/semver' => array('10.4.13beta.2', '10.4.13.0-beta2'),
  72. 'parses long/semver2' => array('v1.13.11-beta.0', '1.13.11.0-beta0'),
  73. 'parses long/semver3' => array('1.13.11.0-beta0', '1.13.11.0-beta0'),
  74. 'expand shorthand' => array('10.4.13-b', '10.4.13.0-beta'),
  75. 'expand shorthand/2' => array('10.4.13-b5', '10.4.13.0-beta5'),
  76. 'strips leading v' => array('v1.0.0', '1.0.0.0'),
  77. 'parses dates y-m as classical' => array('2010.01', '2010.01.0.0'),
  78. 'parses dates w/ . as classical' => array('2010.01.02', '2010.01.02.0'),
  79. 'parses dates y.m.Y as classical' => array('2010.1.555', '2010.1.555.0'),
  80. 'parses dates y.m.Y/2 as classical' => array('2010.10.200', '2010.10.200.0'),
  81. 'parses CalVer YYYYMMDD (as MAJOR) versions' => array('20230131.0.0', '20230131.0.0'),
  82. 'parses CalVer YYYYMMDDhhmm (as MAJOR) versions' => array('202301310000.0.0', '202301310000.0.0'),
  83. 'strips v/datetime' => array('v20100102', '20100102'),
  84. 'parses dates no delimiter' => array('20100102', '20100102'),
  85. 'parses dates no delimiter/2' => array('20100102.0', '20100102.0'),
  86. 'parses dates no delimiter/3' => array('20100102.1.0', '20100102.1.0'),
  87. 'parses dates no delimiter/4' => array('20100102.0.3', '20100102.0.3'),
  88. 'parses dates no delimiter/earliest year' => array('100000', '100000'),
  89. 'parses dates w/ - and .' => array('2010-01-02-10-20-30.0.3', '2010.01.02.10.20.30.0.3'),
  90. 'parses dates w/ - and ./2' => array('2010-01-02-10-20-30.5', '2010.01.02.10.20.30.5'),
  91. 'parses dates w/ -' => array('2010-01-02', '2010.01.02'),
  92. 'parses dates w/ .' => array('2012.06.07', '2012.06.07.0'),
  93. 'parses numbers' => array('2010-01-02.5', '2010.01.02.5'),
  94. 'parses dates y.m.Y' => array('2010.1.555', '2010.1.555.0'),
  95. 'parses datetime' => array('20100102-203040', '20100102.203040'),
  96. 'parses date dev' => array('20100102.x-dev', '20100102.9999999.9999999.9999999-dev'),
  97. 'parses datetime dev' => array('20100102.203040.x-dev', '20100102.203040.9999999.9999999-dev'),
  98. 'parses dt+number' => array('20100102203040-10', '20100102203040.10'),
  99. 'parses dt+patch' => array('20100102-203040-p1', '20100102.203040-patch1'),
  100. 'parses dt Ym' => array('201903.0', '201903.0'),
  101. 'parses dt Ym dev' => array('201903.x-dev', '201903.9999999.9999999.9999999-dev'),
  102. 'parses dt Ym+patch' => array('201903.0-p2', '201903.0-patch2'),
  103. 'parses master' => array('dev-master', 'dev-master'),
  104. 'parses master w/o dev' => array('master', 'dev-master'),
  105. 'parses trunk' => array('dev-trunk', 'dev-trunk'),
  106. 'parses branches' => array('1.x-dev', '1.9999999.9999999.9999999-dev'),
  107. 'parses arbitrary' => array('dev-feature-foo', 'dev-feature-foo'),
  108. 'parses arbitrary/2' => array('DEV-FOOBAR', 'dev-FOOBAR'),
  109. 'parses arbitrary/3' => array('dev-feature/foo', 'dev-feature/foo'),
  110. 'parses arbitrary/4' => array('dev-feature+issue-1', 'dev-feature+issue-1'),
  111. 'ignores aliases' => array('dev-master as 1.0.0', 'dev-master'),
  112. 'ignores aliases/2' => array('dev-load-varnish-only-when-used as ^2.0', 'dev-load-varnish-only-when-used'),
  113. 'ignores aliases/3' => array('dev-load-varnish-only-when-used@dev as ^2.0@dev', 'dev-load-varnish-only-when-used'),
  114. 'ignores stability' => array('1.0.0+foo@dev', '1.0.0.0'),
  115. 'ignores stability/2' => array('dev-load-varnish-only-when-used@stable', 'dev-load-varnish-only-when-used'),
  116. 'semver metadata/2' => array('1.0.0-beta.5+foo', '1.0.0.0-beta5'),
  117. 'semver metadata/3' => array('1.0.0+foo', '1.0.0.0'),
  118. 'semver metadata/4' => array('1.0.0-alpha.3.1+foo', '1.0.0.0-alpha3.1'),
  119. 'semver metadata/5' => array('1.0.0-alpha2.1+foo', '1.0.0.0-alpha2.1'),
  120. 'semver metadata/6' => array('1.0.0-alpha-2.1-3+foo', '1.0.0.0-alpha2.1-3'),
  121. // not supported for BC 'semver metadata/7' => array('1.0.0-0.3.7', '1.0.0.0-0.3.7'),
  122. // not supported for BC 'semver metadata/8' => array('1.0.0-x.7.z.92', '1.0.0.0-x.7.z.92'),
  123. 'metadata w/ alias' => array('1.0.0+foo as 2.0', '1.0.0.0'),
  124. 'keep zero-padding' => array('00.01.03.04', '00.01.03.04'),
  125. 'keep zero-padding/2' => array('000.001.003.004', '000.001.003.004'),
  126. 'keep zero-padding/3' => array('0.000.103.204', '0.000.103.204'),
  127. 'keep zero-padding/4' => array('0700', '0700.0.0.0'),
  128. 'keep zero-padding/5' => array('041.x-dev', '041.9999999.9999999.9999999-dev'),
  129. 'keep zero-padding/6' => array('dev-041.003', 'dev-041.003'),
  130. 'dev with mad name' => array('dev-1.0.0-dev<1.0.5-dev', 'dev-1.0.0-dev<1.0.5-dev'),
  131. 'dev prefix with spaces' => array('dev-foo bar', 'dev-foo bar'),
  132. 'space padding' => array(' 1.0.0', '1.0.0.0'),
  133. 'space padding/2' => array('1.0.0 ', '1.0.0.0'),
  134. );
  135. }
  136. /**
  137. * @dataProvider failingNormalizedVersions
  138. * @param string $input
  139. */
  140. public function testNormalizeFails($input)
  141. {
  142. $this->doExpectException('UnexpectedValueException');
  143. $parser = new VersionParser();
  144. $parser->normalize($input);
  145. }
  146. /**
  147. * @return array<mixed>
  148. */
  149. public function failingNormalizedVersions()
  150. {
  151. return array(
  152. 'empty ' => array(''),
  153. 'invalid chars' => array('a'),
  154. 'invalid type' => array('1.0.0-meh'),
  155. 'too many bits' => array('1.0.0.0.0'),
  156. 'non-dev arbitrary' => array('feature-foo'),
  157. 'metadata w/ space' => array('1.0.0+foo bar'),
  158. 'maven style release' => array('1.0.1-SNAPSHOT'),
  159. 'dev with less than' => array('1.0.0<1.0.5-dev'),
  160. 'dev with less than/2' => array('1.0.0-dev<1.0.5-dev'),
  161. 'dev suffix with spaces' => array('foo bar-dev'),
  162. 'any with spaces' => array('1.0 .2'),
  163. 'no version, no alias' => array(' as '),
  164. 'no version, only alias' => array(' as 1.2'),
  165. 'just an operator' => array('^'),
  166. 'just an operator/2' => array('^8 || ^'),
  167. 'just an operator/3' => array('~'),
  168. 'just an operator/4' => array('~1 ~'),
  169. 'constraint' => array('~1'),
  170. 'constraint/2' => array('^1'),
  171. 'constraint/3' => array('1.*'),
  172. 'date versions with 4 bits' => array('20100102.0.3.4', '20100102.0.3.4'),
  173. 'date versions with 4 bits/earliest year' => array('100000.0.0.0', '100000.0.0.0'),
  174. 'invalid CalVer (as MAJOR) versions/YYYYMMD' => array('2023013.0.0', '2023013.0.0'),
  175. 'invalid CalVer (as MAJOR) versions/YYYYMMDDh' => array('202301311.0.0', '202301311.0.0'),
  176. 'invalid CalVer (as MAJOR) versions/YYYYMMDDhhm' => array('20230131000.0.0', '20230131000.0.0'),
  177. 'invalid CalVer (as MAJOR) versions/YYYYMMDDhhmmX' => array('2023013100000.0.0', '2023013100000.0.0'),
  178. );
  179. }
  180. /**
  181. * @dataProvider failingNormalizedVersionsWithBadAlias
  182. * @param string $fullInput
  183. */
  184. public function testNormalizeFailsAndReportsAliasIssue($fullInput)
  185. {
  186. preg_match('{^([^,\s#]+)(?:#[^ ]+)? +as +([^,\s]+)$}', $fullInput, $match);
  187. $parser = new VersionParser();
  188. $parser->normalize($match[1], $fullInput);
  189. try {
  190. $parser->normalize($match[2], $fullInput);
  191. } catch (\UnexpectedValueException $e) {
  192. $this->assertEquals('Invalid version string "'.$match[2].'" in "'.$fullInput.'", the alias must be an exact version', $e->getMessage());
  193. }
  194. }
  195. /**
  196. * @return array<mixed>
  197. */
  198. public function failingNormalizedVersionsWithBadAlias()
  199. {
  200. return array(
  201. 'Alias and caret' => array('1.0.0+foo as ^2.0'),
  202. 'Alias and tilde' => array('1.0.0+foo as ~2.0'),
  203. 'Alias and greater than' => array('1.0.0+foo as >2.0'),
  204. 'Alias and less than' => array('1.0.0+foo as <2.0'),
  205. 'Bad alias with stability' => array('1.0.0+foo@dev as <2.0@dev'),
  206. );
  207. }
  208. /**
  209. * @dataProvider failingNormalizedVersionsWithBadAliasee
  210. * @param string $fullInput
  211. */
  212. public function testNormalizeFailsAndReportsAliaseeIssue($fullInput)
  213. {
  214. preg_match('{^([^,\s#]+)(?:#[^ ]+)? +as +([^,\s]+)$}', $fullInput, $match);
  215. $parser = new VersionParser();
  216. try {
  217. $parser->normalize($match[1], $fullInput);
  218. } catch (\UnexpectedValueException $e) {
  219. $this->assertEquals('Invalid version string "'.$match[1].'" in "'.$fullInput.'", the alias source must be an exact version, if it is a branch name you should prefix it with dev-', $e->getMessage());
  220. }
  221. $parser->normalize($match[2], $fullInput);
  222. }
  223. /**
  224. * @return array<mixed>
  225. */
  226. public function failingNormalizedVersionsWithBadAliasee()
  227. {
  228. return array(
  229. 'Alias and caret' => array('^2.0 as 1.0.0+foo'),
  230. 'Alias and tilde' => array('~2.0 as 1.0.0+foo'),
  231. 'Alias and greater than' => array('>2.0 as 1.0.0+foo'),
  232. 'Alias and less than' => array('<2.0 as 1.0.0+foo'),
  233. 'Bad aliasee with stability' => array('<2.0@dev as 1.2.3@dev'),
  234. );
  235. }
  236. /**
  237. * @dataProvider successfulNormalizedBranches
  238. * @param string $input
  239. * @param string $expected
  240. */
  241. public function testNormalizeBranch($input, $expected)
  242. {
  243. $parser = new VersionParser();
  244. $this->assertSame((string) $expected, (string) $parser->normalizeBranch($input));
  245. }
  246. /**
  247. * @return array<mixed>
  248. */
  249. public function successfulNormalizedBranches()
  250. {
  251. return array(
  252. 'parses x' => array('v1.x', '1.9999999.9999999.9999999-dev'),
  253. 'parses *' => array('v1.*', '1.9999999.9999999.9999999-dev'),
  254. 'parses digits' => array('v1.0', '1.0.9999999.9999999-dev'),
  255. 'parses digits/2' => array('2.0', '2.0.9999999.9999999-dev'),
  256. 'parses long x' => array('v1.0.x', '1.0.9999999.9999999-dev'),
  257. 'parses long *' => array('v1.0.3.*', '1.0.3.9999999-dev'),
  258. 'parses long digits' => array('v2.4.0', '2.4.0.9999999-dev'),
  259. 'parses long digits/2' => array('2.4.4', '2.4.4.9999999-dev'),
  260. 'parses master' => array('master', 'dev-master'),
  261. 'parses trunk' => array('trunk', 'dev-trunk'),
  262. 'parses arbitrary' => array('feature-a', 'dev-feature-a'),
  263. 'parses arbitrary/2' => array('FOOBAR', 'dev-FOOBAR'),
  264. 'parses arbitrary/3' => array('feature+issue-1', 'dev-feature+issue-1'),
  265. );
  266. }
  267. public function testParseConstraintsIgnoresStabilityFlag()
  268. {
  269. $parser = new VersionParser();
  270. $this->assertSame((string) new Constraint('=', '1.0.0.0'), (string) $parser->parseConstraints('1.0@dev'));
  271. $this->assertSame((string) new Constraint('>=', '1.0.0.0-beta'), (string) $parser->parseConstraints('>=1.0@beta'));
  272. $this->assertSame((string) new Constraint('=', 'dev-load-varnish-only-when-used'), (string) $parser->parseConstraints('dev-load-varnish-only-when-used as ^2.0@dev'));
  273. $this->assertSame((string) new Constraint('=', 'dev-load-varnish-only-when-used'), (string) $parser->parseConstraints('dev-load-varnish-only-when-used@dev as ^2.0@dev'));
  274. }
  275. public function testParseConstraintsIgnoresReferenceOnDevVersion()
  276. {
  277. $parser = new VersionParser();
  278. $this->assertSame((string) new Constraint('=', '1.0.9999999.9999999-dev'), (string) $parser->parseConstraints('1.0.x-dev#abcd123'));
  279. $this->assertSame((string) new Constraint('=', '1.0.9999999.9999999-dev'), (string) $parser->parseConstraints('1.0.x-dev#trunk/@123'));
  280. }
  281. public function testParseConstraintsFailsOnBadReference()
  282. {
  283. $this->doExpectException('UnexpectedValueException');
  284. $parser = new VersionParser();
  285. $this->assertSame((string) new Constraint('=', '1.0.0.0'), (string) $parser->parseConstraints('1.0#abcd123'));
  286. $this->assertSame((string) new Constraint('=', '1.0.0.0'), (string) $parser->parseConstraints('1.0#trunk/@123'));
  287. }
  288. public function testParseConstraintsNudgesRubyDevsTowardsThePathOfRighteousness()
  289. {
  290. $this->doExpectException('UnexpectedValueException', 'Invalid operator "~>", you probably meant to use the "~" operator');
  291. $parser = new VersionParser();
  292. $parser->parseConstraints('~>1.2');
  293. }
  294. /**
  295. * @dataProvider simpleConstraints
  296. *
  297. * @param string $input
  298. * @param Constraint $expected
  299. */
  300. public function testParseConstraintsSimple($input, $expected)
  301. {
  302. $parser = new VersionParser();
  303. $this->assertSame((string) $expected, (string) $parser->parseConstraints($input));
  304. }
  305. /**
  306. * @return array<mixed>
  307. */
  308. public function simpleConstraints()
  309. {
  310. return array(
  311. 'match any' => array('*', new MatchAllConstraint()),
  312. 'match any/v' => array('v*', new Constraint('>=', '0.0.0.0-dev')),
  313. 'match any/2' => array('*.*', new Constraint('>=', '0.0.0.0-dev')),
  314. 'match any/2v' => array('v*.*', new Constraint('>=', '0.0.0.0-dev')),
  315. 'match any/3' => array('*.x.*', new Constraint('>=', '0.0.0.0-dev')),
  316. 'match any/4' => array('x.X.x.*', new Constraint('>=', '0.0.0.0-dev')),
  317. 'not equal' => array('<>1.0.0', new Constraint('<>', '1.0.0.0')),
  318. 'not equal/2' => array('!=1.0.0', new Constraint('!=', '1.0.0.0')),
  319. 'greater than' => array('>1.0.0', new Constraint('>', '1.0.0.0')),
  320. 'lesser than' => array('<1.2.3.4', new Constraint('<', '1.2.3.4-dev')),
  321. 'less/eq than' => array('<=1.2.3', new Constraint('<=', '1.2.3.0')),
  322. 'great/eq than' => array('>=1.2.3', new Constraint('>=', '1.2.3.0-dev')),
  323. 'equals' => array('=1.2.3', new Constraint('=', '1.2.3.0')),
  324. 'double equals' => array('==1.2.3', new Constraint('=', '1.2.3.0')),
  325. 'no op means eq' => array('1.2.3', new Constraint('=', '1.2.3.0')),
  326. 'completes version' => array('=1.0', new Constraint('=', '1.0.0.0')),
  327. 'shorthand beta' => array('1.2.3b5', new Constraint('=', '1.2.3.0-beta5')),
  328. 'shorthand alpha' => array('1.2.3a1', new Constraint('=', '1.2.3.0-alpha1')),
  329. 'shorthand patch' => array('1.2.3p1234', new Constraint('=', '1.2.3.0-patch1234')),
  330. 'shorthand patch/2' => array('1.2.3pl1234', new Constraint('=', '1.2.3.0-patch1234')),
  331. 'accepts spaces' => array('>= 1.2.3', new Constraint('>=', '1.2.3.0-dev')),
  332. 'accepts spaces/2' => array('< 1.2.3', new Constraint('<', '1.2.3.0-dev')),
  333. 'accepts spaces/3' => array('> 1.2.3', new Constraint('>', '1.2.3.0')),
  334. 'accepts master' => array('>=dev-master', new Constraint('>=', 'dev-master')),
  335. 'accepts master/2' => array('dev-master', new Constraint('=', 'dev-master')),
  336. 'accepts arbitrary' => array('dev-feature-a', new Constraint('=', 'dev-feature-a')),
  337. 'regression #550' => array('dev-some-fix', new Constraint('=', 'dev-some-fix')),
  338. 'regression #935' => array('dev-CAPS', new Constraint('=', 'dev-CAPS')),
  339. 'ignores aliases' => array('dev-master as 1.0.0', new Constraint('=', 'dev-master')),
  340. 'lesser than override' => array('<1.2.3.4-stable', new Constraint('<', '1.2.3.4')),
  341. 'great/eq than override' => array('>=1.2.3.4-stable', new Constraint('>=', '1.2.3.4')),
  342. );
  343. }
  344. /**
  345. * @dataProvider wildcardConstraints
  346. *
  347. * @param string $input
  348. * @param Constraint|null $min
  349. * @param Constraint $max
  350. */
  351. public function testParseConstraintsWildcard($input, $min, $max)
  352. {
  353. $parser = new VersionParser();
  354. if ($min) {
  355. $expected = new MultiConstraint(array($min, $max));
  356. } else {
  357. $expected = $max;
  358. }
  359. $this->assertSame((string) $expected, (string) $parser->parseConstraints($input));
  360. }
  361. /**
  362. * @return array<mixed>
  363. */
  364. public function wildcardConstraints()
  365. {
  366. return array(
  367. array('v2.*', new Constraint('>=', '2.0.0.0-dev'), new Constraint('<', '3.0.0.0-dev')),
  368. array('2.*.*', new Constraint('>=', '2.0.0.0-dev'), new Constraint('<', '3.0.0.0-dev')),
  369. array('20.*', new Constraint('>=', '20.0.0.0-dev'), new Constraint('<', '21.0.0.0-dev')),
  370. array('20.*.*', new Constraint('>=', '20.0.0.0-dev'), new Constraint('<', '21.0.0.0-dev')),
  371. array('2.0.*', new Constraint('>=', '2.0.0.0-dev'), new Constraint('<', '2.1.0.0-dev')),
  372. array('2.x', new Constraint('>=', '2.0.0.0-dev'), new Constraint('<', '3.0.0.0-dev')),
  373. array('2.x.x', new Constraint('>=', '2.0.0.0-dev'), new Constraint('<', '3.0.0.0-dev')),
  374. array('2.2.x', new Constraint('>=', '2.2.0.0-dev'), new Constraint('<', '2.3.0.0-dev')),
  375. array('2.10.X', new Constraint('>=', '2.10.0.0-dev'), new Constraint('<', '2.11.0.0-dev')),
  376. array('2.1.3.*', new Constraint('>=', '2.1.3.0-dev'), new Constraint('<', '2.1.4.0-dev')),
  377. array('0.*', null, new Constraint('<', '1.0.0.0-dev')),
  378. array('0.*.*', null, new Constraint('<', '1.0.0.0-dev')),
  379. array('0.x', null, new Constraint('<', '1.0.0.0-dev')),
  380. array('0.x.x', null, new Constraint('<', '1.0.0.0-dev')),
  381. );
  382. }
  383. /**
  384. * @dataProvider tildeConstraints
  385. *
  386. * @param string $input
  387. * @param Constraint|null $min
  388. * @param Constraint $max
  389. */
  390. public function testParseTildeWildcard($input, $min, $max)
  391. {
  392. $parser = new VersionParser();
  393. if ($min) {
  394. $expected = new MultiConstraint(array($min, $max));
  395. } else {
  396. $expected = $max;
  397. }
  398. $this->assertSame((string) $expected, (string) $parser->parseConstraints($input));
  399. }
  400. /**
  401. * @return array<mixed>
  402. */
  403. public function tildeConstraints()
  404. {
  405. return array(
  406. array('~v1', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  407. array('~1.0', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  408. array('~1.0.0', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '1.1.0.0-dev')),
  409. array('~1.2', new Constraint('>=', '1.2.0.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  410. array('~1.2.3', new Constraint('>=', '1.2.3.0-dev'), new Constraint('<', '1.3.0.0-dev')),
  411. array('~1.2.3.4', new Constraint('>=', '1.2.3.4-dev'), new Constraint('<', '1.2.4.0-dev')),
  412. array('~1.2-beta',new Constraint('>=', '1.2.0.0-beta'), new Constraint('<', '2.0.0.0-dev')),
  413. array('~1.2-b2', new Constraint('>=', '1.2.0.0-beta2'), new Constraint('<', '2.0.0.0-dev')),
  414. array('~1.2-BETA2', new Constraint('>=', '1.2.0.0-beta2'), new Constraint('<', '2.0.0.0-dev')),
  415. array('~1.2.2-dev', new Constraint('>=', '1.2.2.0-dev'), new Constraint('<', '1.3.0.0-dev')),
  416. array('~1.2.2-stable', new Constraint('>=', '1.2.2.0'), new Constraint('<', '1.3.0.0-dev')),
  417. array('~201903.0', new Constraint('>=', '201903.0-dev'), new Constraint('<', '201904.0.0.0-dev')),
  418. array('~201903.0-beta', new Constraint('>=', '201903.0-beta'), new Constraint('<', '201904.0.0.0-dev')),
  419. array('~201903.0-stable', new Constraint('>=', '201903.0'), new Constraint('<', '201904.0.0.0-dev')),
  420. array('~201903.205830.1-stable', new Constraint('>=', '201903.205830.1'), new Constraint('<', '201903.205831.0.0-dev')),
  421. array('~2.x-dev', new Constraint('>=', '2.9999999.9999999.9999999-dev'), new Constraint('<', '3.0.0.0-dev')),
  422. array('~2.0.x-dev', new Constraint('>=', '2.0.9999999.9999999-dev'), new Constraint('<', '2.1.0.0-dev')),
  423. array('~2.0.3.x-dev', new Constraint('>=', '2.0.3.9999999-dev'), new Constraint('<', '2.0.4.0-dev')),
  424. array('~0.x-dev', new Constraint('>=', '0.9999999.9999999.9999999-dev'), new Constraint('<', '1.0.0.0-dev')),
  425. );
  426. }
  427. /**
  428. * @dataProvider caretConstraints
  429. *
  430. * @param string $input
  431. * @param Constraint|null $min
  432. * @param Constraint $max
  433. */
  434. public function testParseCaretWildcard($input, $min, $max)
  435. {
  436. $parser = new VersionParser();
  437. if ($min) {
  438. $expected = new MultiConstraint(array($min, $max));
  439. } else {
  440. $expected = $max;
  441. }
  442. $this->assertSame((string) $expected, (string) $parser->parseConstraints($input));
  443. }
  444. /**
  445. * @return array<mixed>
  446. */
  447. public function caretConstraints()
  448. {
  449. return array(
  450. array('^v1', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  451. array('^0', new Constraint('>=', '0.0.0.0-dev'), new Constraint('<', '1.0.0.0-dev')),
  452. array('^0.0', new Constraint('>=', '0.0.0.0-dev'), new Constraint('<', '0.1.0.0-dev')),
  453. array('^1.2', new Constraint('>=', '1.2.0.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  454. array('^1.2.3-beta.2', new Constraint('>=', '1.2.3.0-beta2'), new Constraint('<', '2.0.0.0-dev')),
  455. array('^1.2.3.4', new Constraint('>=', '1.2.3.4-dev'), new Constraint('<', '2.0.0.0-dev')),
  456. array('^1.2.3', new Constraint('>=', '1.2.3.0-dev'), new Constraint('<', '2.0.0.0-dev')),
  457. array('^0.2.3', new Constraint('>=', '0.2.3.0-dev'), new Constraint('<', '0.3.0.0-dev')),
  458. array('^0.2', new Constraint('>=', '0.2.0.0-dev'), new Constraint('<', '0.3.0.0-dev')),
  459. array('^0.2.0', new Constraint('>=', '0.2.0.0-dev'), new Constraint('<', '0.3.0.0-dev')),
  460. array('^0.0.3', new Constraint('>=', '0.0.3.0-dev'), new Constraint('<', '0.0.4.0-dev')),
  461. array('^0.0.3-alpha', new Constraint('>=', '0.0.3.0-alpha'), new Constraint('<', '0.0.4.0-dev')),
  462. array('^0.0.3-dev', new Constraint('>=', '0.0.3.0-dev'), new Constraint('<', '0.0.4.0-dev')),
  463. array('^0.0.3-stable', new Constraint('>=', '0.0.3.0'), new Constraint('<', '0.0.4.0-dev')),
  464. array('^201903.0', new Constraint('>=', '201903.0-dev'), new Constraint('<', '201904.0.0.0-dev')),
  465. array('^201903.0-beta', new Constraint('>=', '201903.0-beta'), new Constraint('<', '201904.0.0.0-dev')),
  466. array('^201903.205830.1-stable', new Constraint('>=', '201903.205830.1'), new Constraint('<', '201904.0.0.0-dev')),
  467. array('^2.x-dev', new Constraint('>=', '2.9999999.9999999.9999999-dev'), new Constraint('<', '3.0.0.0-dev')),
  468. array('^2.0.*-dev', new Constraint('>=', '2.0.9999999.9999999-dev'), new Constraint('<', '3.0.0.0-dev')),
  469. array('^2.0.x-dev', new Constraint('>=', '2.0.9999999.9999999-dev'), new Constraint('<', '3.0.0.0-dev')),
  470. array('^2.0.3.x-dev', new Constraint('>=', '2.0.3.9999999-dev'), new Constraint('<', '3.0.0.0-dev')),
  471. array('^0.x-dev', new Constraint('>=', '0.9999999.9999999.9999999-dev'), new Constraint('<', '1.0.0.0-dev')),
  472. );
  473. }
  474. /**
  475. * @dataProvider hyphenConstraints
  476. *
  477. * @param string $input
  478. * @param Constraint|null $min
  479. * @param Constraint $max
  480. */
  481. public function testParseHyphen($input, $min, $max)
  482. {
  483. $parser = new VersionParser();
  484. if ($min) {
  485. $expected = new MultiConstraint(array($min, $max));
  486. } else {
  487. $expected = $max;
  488. }
  489. $this->assertSame((string) $expected, (string) $parser->parseConstraints($input));
  490. }
  491. /**
  492. * @return array<mixed>
  493. */
  494. public function hyphenConstraints()
  495. {
  496. return array(
  497. array('v1 - v2', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '3.0.0.0-dev')),
  498. array('1.2.3 - 2.3.4.5', new Constraint('>=', '1.2.3.0-dev'), new Constraint('<=', '2.3.4.5')),
  499. array('1.2-beta - 2.3', new Constraint('>=', '1.2.0.0-beta'), new Constraint('<', '2.4.0.0-dev')),
  500. array('1.2-beta - 2.3-dev', new Constraint('>=', '1.2.0.0-beta'), new Constraint('<=', '2.3.0.0-dev')),
  501. array('1.2-RC - 2.3.1', new Constraint('>=', '1.2.0.0-RC'), new Constraint('<=', '2.3.1.0')),
  502. array('1.2.3-alpha - 2.3-RC', new Constraint('>=', '1.2.3.0-alpha'), new Constraint('<=', '2.3.0.0-RC')),
  503. array('1 - 2.0', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '2.1.0.0-dev')),
  504. array('1 - 2.1', new Constraint('>=', '1.0.0.0-dev'), new Constraint('<', '2.2.0.0-dev')),
  505. array('1.2 - 2.1.0', new Constraint('>=', '1.2.0.0-dev'), new Constraint('<=', '2.1.0.0')),
  506. array('1.3 - 2.1.3', new Constraint('>=', '1.3.0.0-dev'), new Constraint('<=', '2.1.3.0')),
  507. array('2.0.3.x-dev - 3.0.3.x-dev', new Constraint('>=', '2.0.3.9999999-dev'), new Constraint('<=', '3.0.3.9999999-dev')),
  508. array('2.0.x-dev - 3.0.x-dev', new Constraint('>=', '2.0.9999999.9999999-dev'), new Constraint('<=', '3.0.9999999.9999999-dev')),
  509. array('2.x-dev - 3.x-dev', new Constraint('>=', '2.9999999.9999999.9999999-dev'), new Constraint('<=', '3.9999999.9999999.9999999-dev')),
  510. array('0.x-dev - 1.x-dev', new Constraint('>=', '0.9999999.9999999.9999999-dev'), new Constraint('<=', '1.9999999.9999999.9999999-dev')),
  511. );
  512. }
  513. /**
  514. * @dataProvider constraintProvider
  515. * @param string $constraint
  516. * @param string $expected
  517. */
  518. public function testParseConstraints($constraint, $expected)
  519. {
  520. $parser = new VersionParser();
  521. $this->assertSame($expected, (string) $parser->parseConstraints($constraint));
  522. }
  523. /**
  524. * @return array<mixed>
  525. */
  526. public function constraintProvider()
  527. {
  528. return array(
  529. // numeric branch
  530. array('3.x-dev', '== 3.9999999.9999999.9999999-dev'),
  531. array('3-dev', '== 3.0.0.0-dev'),
  532. // non-numeric branches
  533. array('dev-3.x', '== dev-3.x'),
  534. array('xsd2php-dev', '== dev-xsd2php'),
  535. array('3.next-dev', '== dev-3.next'),
  536. array('foobar-dev', '== dev-foobar'),
  537. array('dev-xsd2php', '== dev-xsd2php'),
  538. array('dev-3.next', '== dev-3.next'),
  539. array('dev-foobar', '== dev-foobar'),
  540. array('dev-1.0.0-dev<1.0.5-dev', '== dev-1.0.0-dev<1.0.5-dev'),
  541. array('dev-1.0.0-dev<1.0.5', '== dev-1.0.0-dev<1.0.5'),
  542. array('foobar-dev as 2.1.0', '== dev-foobar'),
  543. array('foobar-dev as 2.1.0 || 3.5', '[== dev-foobar || == 3.5.0.0]'),
  544. array('foobar-dev as 2.1.0 || 3.5 as 1.5', '[== dev-foobar || == 3.5.0.0]'),
  545. array('2.1.0 - 2.3-dev', '[>= 2.1.0.0-dev <= 2.3.0.0-dev]'),
  546. array('1.0 - 2.0.x-dev', '[>= 1.0.0.0-dev <= 2.0.9999999.9999999-dev]'),
  547. // borked typo constraints but so common historically that we gotta keep them working
  548. array('^1.', '[>= 1.0.0.0-dev < 2.0.0.0-dev]'),
  549. array('~1.', '[>= 1.0.0.0-dev < 2.0.0.0-dev]'),
  550. array('1.2.', '== 1.2.0.0'),
  551. array('1.2..dev', '== 1.2.0.0-dev'),
  552. array('1.2-.dev', '== 1.2.0.0-dev'),
  553. array('1.2_-dev', '== 1.2.0.0-dev'),
  554. // complex constraints
  555. array('~2.5.9|~2.6,>=2.6.2', '[[>= 2.5.9.0-dev < 2.6.0.0-dev] || [>= 2.6.0.0-dev < 3.0.0.0-dev >= 2.6.2.0-dev]]'),
  556. );
  557. }
  558. /**
  559. * @dataProvider multiConstraintProvider
  560. * @param string $constraint
  561. */
  562. public function testParseConstraintsMulti($constraint)
  563. {
  564. $parser = new VersionParser();
  565. $first = new Constraint('>', '2.0.0.0');
  566. $second = new Constraint('<=', '3.0.0.0');
  567. $multi = new MultiConstraint(array($first, $second));
  568. $this->assertSame((string) $multi, (string) $parser->parseConstraints($constraint));
  569. }
  570. /**
  571. * @return array<mixed>
  572. */
  573. public function multiConstraintProvider()
  574. {
  575. return array(
  576. array('>2.0,<=3.0'),
  577. array('>2.0 <=3.0'),
  578. array('>2.0 <=3.0'),
  579. array('>2.0, <=3.0'),
  580. array('>2.0 ,<=3.0'),
  581. array('>2.0 , <=3.0'),
  582. array('>2.0 , <=3.0'),
  583. array('> 2.0 <= 3.0'),
  584. array('> 2.0 , <= 3.0'),
  585. array(' > 2.0 , <= 3.0 '),
  586. );
  587. }
  588. public function testParseConstraintsMultiWithStabilitySuffix()
  589. {
  590. $parser = new VersionParser();
  591. $first = new Constraint('>=', '1.1.0.0-alpha4');
  592. $second = new Constraint('<', '1.2.9999999.9999999-dev');
  593. $multi = new MultiConstraint(array($first, $second));
  594. $this->assertSame((string) $multi, (string) $parser->parseConstraints('>=1.1.0-alpha4,<1.2.x-dev'));
  595. $first = new Constraint('>=', '1.1.0.0-alpha4');
  596. $second = new Constraint('<', '1.2.0.0-beta2');
  597. $multi = new MultiConstraint(array($first, $second));
  598. $this->assertSame((string) $multi, (string) $parser->parseConstraints('>=1.1.0-alpha4,<1.2-beta2'));
  599. }
  600. /**
  601. * @dataProvider multiConstraintProvider2
  602. *
  603. * @param string $constraint
  604. */
  605. public function testParseConstraintsMultiDisjunctiveHasPrioOverConjuctive($constraint)
  606. {
  607. $parser = new VersionParser();
  608. $first = new Constraint('>', '2.0.0.0');
  609. $second = new Constraint('<', '2.0.5.0-dev');
  610. $third = new Constraint('>', '2.0.6.0');
  611. $multi1 = new MultiConstraint(array($first, $second));
  612. $multi2 = new MultiConstraint(array($multi1, $third), false);
  613. $this->assertSame((string) $multi2, (string) $parser->parseConstraints($constraint));
  614. }
  615. /**
  616. * @return array<mixed>
  617. */
  618. public function multiConstraintProvider2()
  619. {
  620. return array(
  621. array('>2.0,<2.0.5 | >2.0.6'),
  622. array('>2.0,<2.0.5 || >2.0.6'),
  623. array('> 2.0 , <2.0.5 | > 2.0.6'),
  624. );
  625. }
  626. public function testParseConstraintsMultiWithStabilities()
  627. {
  628. $parser = new VersionParser();
  629. $first = new Constraint('>', '2.0.0.0');
  630. $second = new Constraint('<=', '3.0.0.0-dev');
  631. $multi = new MultiConstraint(array($first, $second));
  632. $this->assertSame((string) $multi, (string) $parser->parseConstraints('>2.0@stable,<=3.0@dev'));
  633. }
  634. public function testParseConstraintsMultiWithStabilitiesWildcard()
  635. {
  636. $parser = new VersionParser();
  637. $first = new Constraint('>', '2.0.0.0');
  638. $second = new MatchAllConstraint();
  639. $multi = new MultiConstraint(array($first, $second));
  640. $this->assertSame((string) $multi, (string) $parser->parseConstraints('>2.0@stable,@dev'));
  641. }
  642. public function testParseConstraintsMultiWithStabilitiesZero()
  643. {
  644. $parser = new VersionParser();
  645. $first = new Constraint('>', '2.0.0.0');
  646. $second = new Constraint('==', '0.0.0.0');
  647. $multi = new MultiConstraint(array($first, $second), false);
  648. $this->assertSame((string) $multi, (string) $parser->parseConstraints('>2.0@stable || 0@dev'));
  649. }
  650. /**
  651. * @dataProvider failingConstraints
  652. *
  653. * @param string $input
  654. */
  655. public function testParseConstraintsFails($input)
  656. {
  657. $this->doExpectException('UnexpectedValueException');
  658. $parser = new VersionParser();
  659. $parser->parseConstraints($input);
  660. }
  661. /**
  662. * @return array<mixed>
  663. */
  664. public function failingConstraints()
  665. {
  666. return array(
  667. 'empty ' => array(''),
  668. 'invalid version' => array('1.0.0-meh'),
  669. 'operator abuse' => array('>2.0,,<=3.0'),
  670. 'operator abuse/2' => array('>2.0 ,, <=3.0'),
  671. 'operator abuse/3' => array('>2.0 ||| <=3.0'),
  672. 'leading operator' => array(',^1@dev || ^4@dev'),
  673. 'leading operator/2' => array(',^1@dev'),
  674. 'leading operator/3' => array('|| ^1@dev'),
  675. 'trailing operator' => array('^1@dev ||'),
  676. 'trailing operator/2' => array('^1@dev ,'),
  677. 'caret+wildcard w/o -dev' => array('^2.0.*'),
  678. 'caret+wildcard w/o -dev/2' => array('^2.0.x'),
  679. 'caret+wildcard w/o -dev/3' => array('^2.0.x-beta'),
  680. 'caret+wildcard w/o -dev/4' => array('^2.*'),
  681. 'caret+wildcard w/o -dev/5' => array('^2.x'),
  682. 'caret+wildcard w/o -dev/6' => array('^2.x-beta'),
  683. 'caret+wildcard w/o -dev/7' => array('^2.1.2.*'),
  684. 'caret+wildcard w/o -dev/8' => array('^2.1.2.x'),
  685. 'caret+wildcard w/o -dev/9' => array('^2.1.2.x-beta'),
  686. 'tilde+wildcard w/o -dev' => array('~2.0.*'),
  687. 'tilde+wildcard w/o -dev/2' => array('~2.0.x'),
  688. 'tilde+wildcard w/o -dev/3' => array('~2.0.x-beta'),
  689. 'tilde+wildcard w/o -dev/4' => array('~2.*'),
  690. 'tilde+wildcard w/o -dev/5' => array('~2.x'),
  691. 'tilde+wildcard w/o -dev/6' => array('~2.x-beta'),
  692. 'tilde+wildcard w/o -dev/7' => array('~2.1.2.*'),
  693. 'tilde+wildcard w/o -dev/8' => array('~2.1.2.x'),
  694. 'tilde+wildcard w/o -dev/9' => array('~2.1.2.x-beta'),
  695. 'dash range with wildcard' => array('1.x - 2.*'),
  696. 'dash range with wildcards' => array('2.x.x.x-dev - 3.x.x.x-dev'),
  697. 'broken constraint with dev suffix' => array('^1.*-beta-dev'),
  698. 'broken constraint with dev suffix/2' => array('^1. *-dev'),
  699. 'broken constraint with dev suffix/3' => array('~1.*-beta-dev'),
  700. 'dev suffix conversion only works on simple strings' => array('1.0.0-dev<1.0.5-dev'),
  701. 'dev suffix conversion only works on simple strings/2' => array('*-dev'),
  702. 'just an operator' => array('^'),
  703. 'just an operator/2' => array('^8 || ^'),
  704. 'just an operator/3' => array('~'),
  705. 'just an operator/4' => array('~1 ~'),
  706. );
  707. }
  708. /**
  709. * @dataProvider stabilityProvider
  710. *
  711. * @param string $expected
  712. * @param string $version
  713. */
  714. public function testParseStability($expected, $version)
  715. {
  716. $this->assertSame($expected, VersionParser::parseStability($version));
  717. }
  718. /**
  719. * @return array<mixed>
  720. */
  721. public function stabilityProvider()
  722. {
  723. return array(
  724. array('stable', '1'),
  725. array('stable', '1.0'),
  726. array('stable', '3.2.1'),
  727. array('stable', 'v3.2.1'),
  728. array('dev', 'v2.0.x-dev'),
  729. array('dev', 'v2.0.x-dev#abc123'),
  730. array('dev', 'v2.0.x-dev#trunk/@123'),
  731. array('RC', '3.0-RC2'),
  732. array('dev', 'dev-master'),
  733. array('dev', '3.1.2-dev'),
  734. array('dev', 'dev-feature+issue-1'),
  735. array('stable', '3.1.2-p1'),
  736. array('stable', '3.1.2-pl2'),
  737. array('stable', '3.1.2-patch'),
  738. array('alpha', '3.1.2-alpha5'),
  739. array('beta', '3.1.2-beta'),
  740. array('beta', '2.0B1'),
  741. array('alpha', '1.2.0a1'),
  742. array('alpha', '1.2_a1'),
  743. array('RC', '2.0.0rc1'),
  744. array('alpha', '1.0.0-alpha11+cs-1.1.0'),
  745. array('dev', '1-2_dev'),
  746. );
  747. }
  748. public function testNormalizeStability()
  749. {
  750. $parser = new VersionParser();
  751. $stability = 'rc';
  752. $expectedValue = 'RC';
  753. $result = $parser->normalizeStability($stability);
  754. $this->assertSame($expectedValue, $result);
  755. $stability = 'BeTa';
  756. $expectedValue = 'beta';
  757. $result = $parser->normalizeStability($stability);
  758. $this->assertSame($expectedValue, $result);
  759. }
  760. public function testManipulateVersionStringWithReturnNull()
  761. {
  762. $position = 1;
  763. $increment = 2;
  764. $matches = array(-1, -3, -2, -5, -9);
  765. $parser = new \ReflectionClass('\Composer\Semver\VersionParser');
  766. $manipulateVersionStringMethod = $parser->getMethod('manipulateVersionString');
  767. $manipulateVersionStringMethod->setAccessible(true);
  768. $result = $manipulateVersionStringMethod->invoke(new VersionParser(), $matches, $position, $increment);
  769. $this->assertNull($result);
  770. }
  771. public function testComplexConjunctive()
  772. {
  773. $parser = new VersionParser();
  774. $version = new Constraint('=', '1.0.1.0');
  775. $parsed = $parser->parseConstraints('~0.1 || ~1.0 !=1.0.1');
  776. $this->assertFalse($parsed->matches($version), '"~0.1 || ~1.0 !=1.0.1" should not allow version "1.0.1.0"');
  777. }
  778. /**
  779. * @param class-string $class
  780. * @param string|null $message
  781. * @return void
  782. */
  783. private function doExpectException($class, $message = null)
  784. {
  785. if (method_exists($this, 'expectException')) {
  786. $this->expectException($class);
  787. if ($message) {
  788. $this->expectExceptionMessage($message);
  789. }
  790. } else {
  791. // @phpstan-ignore-next-line
  792. $this->setExpectedException($class, $message);
  793. }
  794. }
  795. }