2011-04-05 15:37:19 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
2011-04-16 12:42:35 +00:00
|
|
|
* This file is part of Composer.
|
2011-04-05 15:37:19 +00:00
|
|
|
*
|
2011-04-16 12:42:35 +00:00
|
|
|
* (c) Nils Adermann <naderman@naderman.de>
|
|
|
|
* Jordi Boggiano <j.boggiano@seld.be>
|
2011-04-05 15:37:19 +00:00
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Composer\Test\DependencyResolver;
|
|
|
|
|
2011-05-23 00:18:11 +00:00
|
|
|
use Composer\Repository\ArrayRepository;
|
2011-07-08 11:09:39 +00:00
|
|
|
use Composer\Repository\PlatformRepository;
|
2011-07-21 11:42:47 +00:00
|
|
|
use Composer\Repository\ComposerRepository;
|
2011-04-05 15:37:19 +00:00
|
|
|
use Composer\DependencyResolver\DefaultPolicy;
|
|
|
|
use Composer\DependencyResolver\Pool;
|
|
|
|
use Composer\DependencyResolver\Request;
|
|
|
|
use Composer\DependencyResolver\Solver;
|
2011-05-23 00:18:11 +00:00
|
|
|
use Composer\Package\MemoryPackage;
|
|
|
|
use Composer\Package\Link;
|
|
|
|
use Composer\Package\LinkConstraint\VersionConstraint;
|
2011-04-05 15:37:19 +00:00
|
|
|
|
|
|
|
class SolverTest extends \PHPUnit_Framework_TestCase
|
|
|
|
{
|
2011-08-05 08:08:21 +00:00
|
|
|
protected $pool;
|
|
|
|
protected $repo;
|
|
|
|
protected $repoInstalled;
|
|
|
|
protected $request;
|
|
|
|
protected $policy;
|
|
|
|
|
|
|
|
public function setUp()
|
2011-04-05 15:37:19 +00:00
|
|
|
{
|
2011-08-05 08:08:21 +00:00
|
|
|
$this->pool = new Pool;
|
|
|
|
$this->repo = new ArrayRepository;
|
|
|
|
$this->repoInstalled = new ArrayRepository;
|
|
|
|
|
|
|
|
$this->request = new Request($this->pool);
|
|
|
|
$this->policy = new DefaultPolicy;
|
|
|
|
$this->solver = new Solver($this->policy, $this->pool, $this->repoInstalled);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverInstallSingle()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverInstallWithDeps()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('B', '1.0'));
|
|
|
|
$this->repo->addPackage($newPackageB = new MemoryPackage('B', '1.1'));
|
2011-04-05 15:37:19 +00:00
|
|
|
|
2011-05-23 00:18:11 +00:00
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('<', '1.1'), 'requires')));
|
2011-04-05 15:37:19 +00:00
|
|
|
|
2011-08-05 08:08:21 +00:00
|
|
|
$this->reposComplete();
|
2011-04-05 15:37:19 +00:00
|
|
|
|
2011-08-05 08:08:21 +00:00
|
|
|
$this->request->install('A');
|
2011-04-05 15:37:19 +00:00
|
|
|
|
2011-08-05 08:08:21 +00:00
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $packageB),
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
2011-04-05 15:37:19 +00:00
|
|
|
|
2011-08-05 08:08:21 +00:00
|
|
|
public function testSolverInstallInstalled()
|
2011-08-20 22:19:47 +00:00
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverInstallInstalledWithAlternative()
|
2011-08-05 08:08:21 +00:00
|
|
|
{
|
|
|
|
$this->repo->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repoInstalled->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverRemoveSingle()
|
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->remove('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'remove', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverRemoveUninstalled()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->remove('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverUpdateSingle()
|
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($newPackageA = new MemoryPackage('A', '1.1'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->update('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
2011-08-20 22:38:31 +00:00
|
|
|
array('job' => 'update', 'from' => $packageA, 'to' => $newPackageA),
|
2011-08-05 08:08:21 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSolverUpdateCurrent()
|
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage(new MemoryPackage('A', '1.0'));
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->update('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array());
|
|
|
|
}
|
|
|
|
|
2011-08-21 10:30:06 +00:00
|
|
|
public function testSolverAllJobs()
|
2011-08-05 08:08:21 +00:00
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage($packageD = new MemoryPackage('D', '1.0'));
|
|
|
|
$this->repoInstalled->addPackage($oldPackageC = new MemoryPackage('C', '1.0'));
|
|
|
|
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '2.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('B', '1.0'));
|
|
|
|
$this->repo->addPackage($newPackageB = new MemoryPackage('B', '1.1'));
|
|
|
|
$this->repo->addPackage($packageC = new MemoryPackage('C', '1.1'));
|
|
|
|
$this->repo->addPackage(new MemoryPackage('D', '1.0'));
|
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('<', '1.1'), 'requires')));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
$this->request->update('C');
|
|
|
|
$this->request->remove('D');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
2011-08-20 22:38:31 +00:00
|
|
|
array('job' => 'update', 'from' => $oldPackageC, 'to' => $packageC),
|
2011-08-05 08:08:21 +00:00
|
|
|
array('job' => 'install', 'package' => $packageB),
|
|
|
|
array('job' => 'remove', 'package' => $packageD),
|
2011-08-20 22:38:31 +00:00
|
|
|
array('job' => 'install', 'package' => $packageA),
|
2011-08-05 08:08:21 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-08-21 10:30:06 +00:00
|
|
|
public function testSolverThreeAlternativeRequireAndConflict()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '2.0'));
|
|
|
|
$this->repo->addPackage($middlePackageB = new MemoryPackage('B', '1.0'));
|
|
|
|
$this->repo->addPackage($newPackageB = new MemoryPackage('B', '1.1'));
|
|
|
|
$this->repo->addPackage($oldPackageB = new MemoryPackage('B', '0.9'));
|
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('<', '1.1'), 'requires')));
|
|
|
|
$packageA->setConflicts(array(new Link('A', 'B', new VersionConstraint('<', '1.0'), 'conflicts')));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $middlePackageB),
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-08-21 11:08:34 +00:00
|
|
|
public function testSolverObsolete()
|
|
|
|
{
|
|
|
|
$this->repoInstalled->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('B', '1.0'));
|
|
|
|
$packageB->setReplaces(array(new Link('B', 'A', null)));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('B');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'update', 'from' => $packageA, 'to' => $packageB),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-08-21 03:05:11 +00:00
|
|
|
public function testInstallOneOfTwoAlternatives()
|
2011-08-05 08:17:07 +00:00
|
|
|
{
|
2011-08-21 03:05:11 +00:00
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('A', '1.0'));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
2011-08-05 08:17:07 +00:00
|
|
|
|
2011-09-09 08:28:50 +00:00
|
|
|
public function testInstallProvider()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageQ = new MemoryPackage('Q', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('B', '0.8'));
|
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('>=', '1.0'), 'requires')));
|
|
|
|
$packageQ->setProvides(array(new Link('Q', 'B', new VersionConstraint('=', '1.0'), 'provides')));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $packageQ),
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testInstallCircularRequire()
|
|
|
|
{
|
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB1 = new MemoryPackage('B', '0.9'));
|
|
|
|
$this->repo->addPackage($packageB2 = new MemoryPackage('B', '1.1'));
|
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('>=', '1.0'), 'requires')));
|
|
|
|
$packageB2->setRequires(array(new Link('B', 'A', new VersionConstraint('>=', '1.0'), 'requires')));
|
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
|
|
|
$this->request->install('A');
|
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
|
|
|
array('job' => 'install', 'package' => $packageB2),
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-09-25 21:50:54 +00:00
|
|
|
public function testInstallAlternativeWithCircularRequire()
|
2011-09-09 08:28:50 +00:00
|
|
|
{
|
2011-09-25 21:50:54 +00:00
|
|
|
$this->markTestIncomplete();
|
2011-09-09 08:28:50 +00:00
|
|
|
|
2011-09-25 21:50:54 +00:00
|
|
|
$this->repo->addPackage($packageA = new MemoryPackage('A', '1.0'));
|
|
|
|
$this->repo->addPackage($packageB = new MemoryPackage('B', '1.0'));
|
|
|
|
$this->repo->addPackage($packageC = new MemoryPackage('C', '1.0'));
|
|
|
|
$this->repo->addPackage($packageD = new MemoryPackage('D', '1.0'));
|
|
|
|
$packageA->setRequires(array(new Link('A', 'B', new VersionConstraint('>=', '1.0'), 'requires')));
|
|
|
|
$packageB->setRequires(array(new Link('B', 'Virtual', new VersionConstraint('>=', '1.0'), 'requires')));
|
|
|
|
$packageC->setRequires(array(new Link('C', 'Virtual', new VersionConstraint('==', '1.0'), 'provides')));
|
|
|
|
$packageD->setRequires(array(new Link('D', 'Virtual', new VersionConstraint('==', '1.0'), 'provides')));
|
2011-09-09 08:28:50 +00:00
|
|
|
|
|
|
|
$this->reposComplete();
|
|
|
|
|
2011-09-25 21:50:54 +00:00
|
|
|
$this->request->install('A');
|
2011-09-09 08:28:50 +00:00
|
|
|
|
|
|
|
$this->checkSolverResult(array(
|
2011-09-25 21:50:54 +00:00
|
|
|
array('job' => 'install', 'package' => $packageC),
|
|
|
|
array('job' => 'install', 'package' => $packageB),
|
|
|
|
array('job' => 'install', 'package' => $packageA),
|
2011-09-09 08:28:50 +00:00
|
|
|
));
|
2011-09-25 21:50:54 +00:00
|
|
|
}
|
2011-09-09 08:28:50 +00:00
|
|
|
|
2011-08-05 08:08:21 +00:00
|
|
|
protected function reposComplete()
|
|
|
|
{
|
|
|
|
$this->pool->addRepository($this->repoInstalled);
|
|
|
|
$this->pool->addRepository($this->repo);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function checkSolverResult(array $expected)
|
|
|
|
{
|
2011-09-23 23:29:22 +00:00
|
|
|
$transaction = $this->solver->solve($this->request);
|
|
|
|
|
|
|
|
$result = array();
|
|
|
|
foreach ($transaction as $operation) {
|
|
|
|
if ('update' === $operation->getJobType()) {
|
2011-09-25 11:40:12 +00:00
|
|
|
$result[] = array(
|
|
|
|
'job' => 'update',
|
|
|
|
'from' => $operation->getInitialPackage(),
|
|
|
|
'to' => $operation->getTargetPackage()
|
|
|
|
);
|
2011-09-23 23:29:22 +00:00
|
|
|
} else {
|
2011-09-25 11:40:12 +00:00
|
|
|
$job = ('uninstall' === $operation->getJobType() ? 'remove' : 'install');
|
|
|
|
$result[] = array(
|
|
|
|
'job' => $job,
|
|
|
|
'package' => $operation->getPackage()
|
|
|
|
);
|
2011-09-23 23:29:22 +00:00
|
|
|
}
|
2011-08-20 22:19:47 +00:00
|
|
|
}
|
|
|
|
|
2011-05-23 00:23:21 +00:00
|
|
|
$this->assertEquals($expected, $result);
|
2011-04-05 15:37:19 +00:00
|
|
|
}
|
2011-07-21 11:42:47 +00:00
|
|
|
|
2011-04-05 15:37:19 +00:00
|
|
|
}
|