1
0
Fork 0
composer/tests/Composer/Test/Plugin/PluginInstallerTest.php

490 lines
18 KiB
PHP
Raw Normal View History

2022-02-23 15:58:18 +00:00
<?php declare(strict_types=1);
2011-11-05 22:51:35 +00:00
/*
* This file is part of Composer.
*
* (c) Nils Adermann <naderman@naderman.de>
* Jordi Boggiano <j.boggiano@seld.be>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Composer\Test\Plugin;
2011-11-05 22:51:35 +00:00
use Composer\Composer;
use Composer\Config;
use Composer\Installer\PluginInstaller;
2022-07-05 14:22:29 +00:00
use Composer\Json\JsonFile;
use Composer\Package\CompleteAliasPackage;
use Composer\Package\CompletePackage;
2011-11-05 22:51:35 +00:00
use Composer\Package\Loader\JsonLoader;
use Composer\Package\Loader\ArrayLoader;
2022-07-05 14:22:29 +00:00
use Composer\Package\Locker;
2021-01-12 13:00:02 +00:00
use Composer\Package\RootPackage;
2013-09-05 12:32:09 +00:00
use Composer\Plugin\PluginManager;
use Composer\IO\BufferIO;
use Composer\EventDispatcher\EventDispatcher;
2013-01-06 19:34:52 +00:00
use Composer\Autoload\AutoloadGenerator;
use Composer\Test\TestCase;
use Composer\Util\Filesystem;
2022-07-05 14:22:29 +00:00
use Composer\Util\Platform;
2011-11-05 22:51:35 +00:00
class PluginInstallerTest extends TestCase
2011-11-05 22:51:35 +00:00
{
/**
* @var Composer
*/
protected $composer;
/**
* @var PluginManager
*/
protected $pm;
/**
* @var AutoloadGenerator
*/
protected $autoloadGenerator;
/**
* @var array<CompletePackage|CompleteAliasPackage>
*/
protected $packages;
/**
* @var string
*/
protected $directory;
/**
* @var \PHPUnit\Framework\MockObject\MockObject&\Composer\Installer\InstallationManager
*/
protected $im;
/**
* @var \PHPUnit\Framework\MockObject\MockObject&\Composer\Repository\InstalledRepositoryInterface
*/
protected $repository;
/**
* @var BufferIO
*/
protected $io;
2021-12-08 16:03:05 +00:00
protected function setUp(): void
2011-11-05 22:51:35 +00:00
{
$loader = new JsonLoader(new ArrayLoader());
2022-08-17 12:20:07 +00:00
$this->packages = [];
$this->directory = self::getUniqueTmpDirectory();
for ($i = 1; $i <= 8; $i++) {
$filename = '/Fixtures/plugin-v'.$i.'/composer.json';
2014-10-17 17:46:01 +00:00
mkdir(dirname($this->directory . $filename), 0777, true);
$this->packages[] = $loader->load(__DIR__ . $filename);
2011-11-05 22:51:35 +00:00
}
$dm = $this->getMockBuilder('Composer\Downloader\DownloadManager')
2011-11-05 22:51:35 +00:00
->disableOriginalConstructor()
->getMock();
2022-02-22 21:10:52 +00:00
$dm->expects($this->any())
->method('install')
->will($this->returnValue(\React\Promise\resolve(null)));
2022-02-22 21:10:52 +00:00
$dm->expects($this->any())
->method('update')
->will($this->returnValue(\React\Promise\resolve(null)));
2022-02-22 21:10:52 +00:00
$dm->expects($this->any())
->method('remove')
->will($this->returnValue(\React\Promise\resolve(null)));
2011-11-05 22:51:35 +00:00
$this->repository = $this->getMockBuilder('Composer\Repository\InstalledRepositoryInterface')->getMock();
2012-01-17 22:13:35 +00:00
$rm = $this->getMockBuilder('Composer\Repository\RepositoryManager')
->disableOriginalConstructor()
->getMock();
$rm->expects($this->any())
2013-03-03 16:18:50 +00:00
->method('getLocalRepository')
->will($this->returnValue($this->repository));
$im = $this->getMockBuilder('Composer\Installer\InstallationManager')->disableOriginalConstructor()->getMock();
$im->expects($this->any())
->method('getInstallPath')
2022-08-17 12:20:07 +00:00
->will($this->returnCallback(static function ($package): string {
return __DIR__.'/Fixtures/'.$package->getPrettyName();
}));
$this->io = new BufferIO();
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')->disableOriginalConstructor()->getMock();
2013-01-06 19:34:52 +00:00
$this->autoloadGenerator = new AutoloadGenerator($dispatcher);
$this->composer = new Composer();
2020-02-07 04:43:57 +00:00
$config = new Config(false);
$this->composer->setConfig($config);
$this->composer->setDownloadManager($dm);
$this->composer->setRepositoryManager($rm);
$this->composer->setInstallationManager($im);
2013-01-06 19:34:52 +00:00
$this->composer->setAutoloadGenerator($this->autoloadGenerator);
$this->composer->setEventDispatcher(new EventDispatcher($this->composer, $this->io));
2021-01-12 13:00:02 +00:00
$this->composer->setPackage(new RootPackage('dummy/root', '1.0.0.0', '1.0.0'));
2022-07-05 14:22:29 +00:00
$this->composer->setLocker(new Locker($this->io, new JsonFile(Platform::getDevNull()), $im, '{}'));
2022-08-17 12:20:07 +00:00
$config->merge([
'config' => [
'vendor-dir' => $this->directory.'/Fixtures/',
'home' => $this->directory.'/Fixtures',
'bin-dir' => $this->directory.'/Fixtures/bin',
'allow-plugins' => true,
2022-08-17 12:20:07 +00:00
],
]);
$this->pm = new PluginManager($this->io, $this->composer);
$this->composer->setPluginManager($this->pm);
2011-11-05 22:51:35 +00:00
}
2021-12-08 16:03:05 +00:00
protected function tearDown(): void
2013-09-22 17:41:54 +00:00
{
parent::tearDown();
$filesystem = new Filesystem();
2013-09-20 04:31:06 +00:00
$filesystem->removeDirectory($this->directory);
}
public function testInstallNewPlugin(): void
2011-11-05 22:51:35 +00:00
{
$this->repository
->expects($this->any())
2011-11-05 22:51:35 +00:00
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([]));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
2011-11-05 22:51:35 +00:00
2012-04-14 13:45:25 +00:00
$installer->install($this->repository, $this->packages[0]);
$plugins = $this->pm->getPlugins();
self::assertEquals('installer-v1', $plugins[0]->version); // @phpstan-ignore property.notFound
self::assertEquals(
'activate v1'.PHP_EOL,
$this->io->getOutput()
);
2011-11-05 22:51:35 +00:00
}
public function testInstallPluginWithRootPackageHavingFilesAutoload(): void
{
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([]));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
$this->autoloadGenerator->setDevMode(true);
2022-08-17 12:20:07 +00:00
$this->composer->getPackage()->setAutoload(['files' => [__DIR__ . '/Fixtures/files_autoload_which_should_not_run.php']]);
$this->composer->getPackage()->setDevAutoload(['files' => [__DIR__ . '/Fixtures/files_autoload_which_should_not_run.php']]);
$installer->install($this->repository, $this->packages[0]);
$plugins = $this->pm->getPlugins();
self::assertEquals(
'activate v1'.PHP_EOL,
$this->io->getOutput()
);
self::assertEquals('installer-v1', $plugins[0]->version); // @phpstan-ignore property.notFound
}
public function testInstallMultiplePlugins(): void
{
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([$this->packages[3]]));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
$installer->install($this->repository, $this->packages[3]);
$plugins = $this->pm->getPlugins();
self::assertEquals('plugin1', $plugins[0]->name); // @phpstan-ignore property.notFound
self::assertEquals('installer-v4', $plugins[0]->version); // @phpstan-ignore property.notFound
self::assertEquals('plugin2', $plugins[1]->name); // @phpstan-ignore property.notFound
self::assertEquals('installer-v4', $plugins[1]->version); // @phpstan-ignore property.notFound
self::assertEquals('activate v4-plugin1'.PHP_EOL.'activate v4-plugin2'.PHP_EOL, $this->io->getOutput());
}
public function testUpgradeWithNewClassName(): void
2011-11-05 22:51:35 +00:00
{
$this->repository
->expects($this->any())
2011-11-05 22:51:35 +00:00
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([$this->packages[0]]));
2011-11-05 22:51:35 +00:00
$this->repository
->expects($this->exactly(2))
2011-11-05 22:51:35 +00:00
->method('hasPackage')
->will($this->onConsecutiveCalls(true, false));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
2011-11-05 22:51:35 +00:00
2012-04-14 13:45:25 +00:00
$installer->update($this->repository, $this->packages[0], $this->packages[1]);
$plugins = $this->pm->getPlugins();
self::assertCount(1, $plugins);
self::assertEquals('installer-v2', $plugins[1]->version); // @phpstan-ignore property.notFound
self::assertEquals('activate v1'.PHP_EOL.'deactivate v1'.PHP_EOL.'activate v2'.PHP_EOL, $this->io->getOutput());
}
public function testUninstall(): void
{
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([$this->packages[0]]));
$this->repository
->expects($this->exactly(1))
->method('hasPackage')
->will($this->onConsecutiveCalls(true, false));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
$installer->uninstall($this->repository, $this->packages[0]);
$plugins = $this->pm->getPlugins();
self::assertCount(0, $plugins);
self::assertEquals('activate v1'.PHP_EOL.'deactivate v1'.PHP_EOL.'uninstall v1'.PHP_EOL, $this->io->getOutput());
2011-11-05 22:51:35 +00:00
}
public function testUpgradeWithSameClassName(): void
2011-11-05 22:51:35 +00:00
{
$this->repository
->expects($this->any())
2011-11-05 22:51:35 +00:00
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([$this->packages[1]]));
2011-11-05 22:51:35 +00:00
$this->repository
->expects($this->exactly(2))
2011-11-05 22:51:35 +00:00
->method('hasPackage')
->will($this->onConsecutiveCalls(true, false));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
2011-11-05 22:51:35 +00:00
2012-04-14 13:45:25 +00:00
$installer->update($this->repository, $this->packages[1], $this->packages[2]);
2011-11-05 22:51:35 +00:00
$plugins = $this->pm->getPlugins();
self::assertEquals('installer-v3', $plugins[1]->version); // @phpstan-ignore property.notFound
self::assertEquals('activate v2'.PHP_EOL.'deactivate v2'.PHP_EOL.'activate v3'.PHP_EOL, $this->io->getOutput());
2011-11-05 22:51:35 +00:00
}
public function testRegisterPluginOnlyOneTime(): void
{
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([]));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
$installer->install($this->repository, $this->packages[0]);
$installer->install($this->repository, clone $this->packages[0]);
$plugins = $this->pm->getPlugins();
self::assertCount(1, $plugins);
self::assertEquals('installer-v1', $plugins[0]->version); // @phpstan-ignore property.notFound
self::assertEquals('activate v1'.PHP_EOL, $this->io->getOutput());
}
/**
* @param array<CompletePackage|CompleteAliasPackage> $plugins
*/
2022-08-17 12:20:07 +00:00
private function setPluginApiVersionWithPlugins(string $newPluginApiVersion, array $plugins = []): void
{
// reset the plugin manager's installed plugins
$this->pm = $this->getMockBuilder('Composer\Plugin\PluginManager')
2022-08-17 12:20:07 +00:00
->onlyMethods(['getPluginApiVersion'])
->setConstructorArgs([$this->io, $this->composer])
->getMock();
// mock the Plugin API version
$this->pm->expects($this->any())
->method('getPluginApiVersion')
->will($this->returnValue($newPluginApiVersion));
$plugApiInternalPackage = self::getPackage(
'composer-plugin-api',
$newPluginApiVersion,
'Composer\Package\CompletePackage'
);
// Add the plugins to the repo along with the internal Plugin package on which they all rely.
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnCallback(static function () use ($plugApiInternalPackage, $plugins): array {
return array_merge([$plugApiInternalPackage], $plugins);
}));
$this->pm->loadInstalledPlugins();
}
public function testStarPluginVersionWorksWithAnyAPIVersion(): void
{
2022-08-17 12:20:07 +00:00
$starVersionPlugin = [$this->packages[4]];
$this->setPluginApiVersionWithPlugins('1.0.0', $starVersionPlugin);
self::assertCount(1, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('1.9.9', $starVersionPlugin);
self::assertCount(1, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('2.0.0-dev', $starVersionPlugin);
self::assertCount(1, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('100.0.0-stable', $starVersionPlugin);
self::assertCount(1, $this->pm->getPlugins());
}
public function testPluginConstraintWorksOnlyWithCertainAPIVersion(): void
{
2022-08-17 12:20:07 +00:00
$pluginWithApiConstraint = [$this->packages[5]];
$this->setPluginApiVersionWithPlugins('1.0.0', $pluginWithApiConstraint);
self::assertCount(0, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('1.1.9', $pluginWithApiConstraint);
self::assertCount(0, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('1.2.0', $pluginWithApiConstraint);
self::assertCount(1, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('1.9.9', $pluginWithApiConstraint);
self::assertCount(1, $this->pm->getPlugins());
}
public function testPluginRangeConstraintsWorkOnlyWithCertainAPIVersion(): void
{
2022-08-17 12:20:07 +00:00
$pluginWithApiConstraint = [$this->packages[6]];
$this->setPluginApiVersionWithPlugins('1.0.0', $pluginWithApiConstraint);
self::assertCount(0, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('3.0.0', $pluginWithApiConstraint);
self::assertCount(1, $this->pm->getPlugins());
$this->setPluginApiVersionWithPlugins('5.5.0', $pluginWithApiConstraint);
self::assertCount(0, $this->pm->getPlugins());
}
public function testCommandProviderCapability(): void
{
$this->repository
->expects($this->any())
->method('getPackages')
2022-08-17 12:20:07 +00:00
->will($this->returnValue([$this->packages[7]]));
$installer = new PluginInstaller($this->io, $this->composer);
$this->pm->loadInstalledPlugins();
/** @var \Composer\Plugin\Capability\CommandProvider[] $caps */
2022-08-17 12:20:07 +00:00
$caps = $this->pm->getPluginCapabilities('Composer\Plugin\Capability\CommandProvider', ['composer' => $this->composer, 'io' => $this->io]);
self::assertCount(1, $caps);
self::assertInstanceOf('Composer\Plugin\Capability\CommandProvider', $caps[0]);
$commands = $caps[0]->getCommands();
self::assertCount(1, $commands);
self::assertInstanceOf('Composer\Command\BaseCommand', $commands[0]);
}
public function testIncapablePluginIsCorrectlyDetected(): void
{
$plugin = $this->getMockBuilder('Composer\Plugin\PluginInterface')
->getMock();
self::assertNull($this->pm->getPluginCapability($plugin, 'Fake\Ability'));
}
public function testCapabilityImplementsComposerPluginApiClassAndIsConstructedWithArgs(): void
{
$capabilityApi = 'Composer\Plugin\Capability\Capability';
$capabilityImplementation = 'Composer\Test\Plugin\Mock\Capability';
$plugin = $this->getMockBuilder('Composer\Test\Plugin\Mock\CapablePluginInterface')
->getMock();
$plugin->expects($this->once())
->method('getCapabilities')
2022-08-17 12:20:07 +00:00
->will($this->returnCallback(static function () use ($capabilityImplementation, $capabilityApi): array {
return [$capabilityApi => $capabilityImplementation];
}));
/** @var \Composer\Test\Plugin\Mock\Capability $capability */
2022-08-17 12:20:07 +00:00
$capability = $this->pm->getPluginCapability($plugin, $capabilityApi, ['a' => 1, 'b' => 2]);
self::assertInstanceOf($capabilityApi, $capability);
self::assertInstanceOf($capabilityImplementation, $capability);
self::assertSame(['a' => 1, 'b' => 2, 'plugin' => $plugin], $capability->args);
}
/** @return mixed[] */
public static function invalidImplementationClassNames(): array
{
2022-08-17 12:20:07 +00:00
return [
[null],
[""],
[0],
[1000],
[" "],
[[1]],
[[]],
[new \stdClass()],
];
}
/**
* @dataProvider invalidImplementationClassNames
2022-02-22 15:47:09 +00:00
* @param mixed $invalidImplementationClassNames
2021-12-09 19:55:26 +00:00
* @param class-string<\Throwable> $expect
*/
2022-02-22 15:47:09 +00:00
public function testQueryingWithInvalidCapabilityClassNameThrows($invalidImplementationClassNames, string $expect = 'UnexpectedValueException'): void
{
2021-12-09 19:55:26 +00:00
self::expectException($expect);
$capabilityApi = 'Composer\Plugin\Capability\Capability';
$plugin = $this->getMockBuilder('Composer\Test\Plugin\Mock\CapablePluginInterface')
->getMock();
$plugin->expects($this->once())
->method('getCapabilities')
2022-08-17 12:20:07 +00:00
->will($this->returnCallback(static function () use ($invalidImplementationClassNames, $capabilityApi): array {
return [$capabilityApi => $invalidImplementationClassNames];
}));
$this->pm->getPluginCapability($plugin, $capabilityApi);
}
public function testQueryingNonProvidedCapabilityReturnsNullSafely(): void
2016-01-22 19:09:44 +00:00
{
$capabilityApi = 'Composer\Plugin\Capability\MadeUpCapability';
$plugin = $this->getMockBuilder('Composer\Test\Plugin\Mock\CapablePluginInterface')
->getMock();
$plugin->expects($this->once())
->method('getCapabilities')
2022-08-17 12:20:07 +00:00
->will($this->returnCallback(static function (): array {
return [];
2016-01-22 19:09:44 +00:00
}));
2021-10-27 14:18:24 +00:00
self::assertNull($this->pm->getPluginCapability($plugin, $capabilityApi));
2016-01-22 19:09:44 +00:00
}
2022-02-22 15:47:09 +00:00
/** @return mixed[] */
public static function nonExistingOrInvalidImplementationClassTypes(): array
2022-02-22 15:47:09 +00:00
{
2022-08-17 12:20:07 +00:00
return [
['\stdClass'],
['NonExistentClassLikeMiddleClass'],
];
2022-02-22 15:47:09 +00:00
}
/**
* @dataProvider nonExistingOrInvalidImplementationClassTypes
*/
2022-02-22 15:47:09 +00:00
public function testQueryingWithNonExistingOrWrongCapabilityClassTypesThrows(string $wrongImplementationClassTypes): void
{
$this->testQueryingWithInvalidCapabilityClassNameThrows($wrongImplementationClassTypes, 'RuntimeException');
}
2012-06-14 10:10:01 +00:00
}