1
0
Fork 0
composer/tests/Composer/Test/EventDispatcher/EventDispatcherTest.php

633 lines
22 KiB
PHP
Raw Normal View History

<?php
/*
* 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\EventDispatcher;
use Composer\EventDispatcher\Event;
2016-07-21 22:20:56 +00:00
use Composer\EventDispatcher\EventDispatcher;
use Composer\Installer\InstallerEvents;
2016-02-10 14:35:53 +00:00
use Composer\Config;
use Composer\Composer;
use Composer\IO\IOInterface;
use Composer\Test\TestCase;
use Composer\IO\BufferIO;
use Composer\Script\ScriptEvents;
2018-11-12 14:57:44 +00:00
use Composer\Script\Event as ScriptEvent;
2012-12-06 08:56:27 +00:00
use Composer\Util\ProcessExecutor;
use Composer\Util\Platform;
use Symfony\Component\Console\Output\OutputInterface;
class EventDispatcherTest extends TestCase
{
public function testListenerExceptionsAreCaught(): void
{
2021-12-09 19:55:26 +00:00
self::expectException('RuntimeException');
$io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock();
$dispatcher = $this->getDispatcherStubForListenersTest(array(
2015-09-28 09:51:14 +00:00
'Composer\Test\EventDispatcher\EventDispatcherTest::call',
), $io);
$io->expects($this->once())
2015-07-04 11:22:58 +00:00
->method('isVerbose')
->willReturn(0);
$io->expects($this->atLeast(2))
->method('writeError')
->withConsecutive(
['> Composer\Test\EventDispatcher\EventDispatcherTest::call'],
['<error>Script Composer\Test\EventDispatcher\EventDispatcherTest::call handling the post-install-cmd event terminated with an exception</error>']
);
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
}
/**
* @dataProvider provideValidCommands
*
* @param string $command
*/
public function testDispatcherCanExecuteSingleCommandLineScript($command): void
{
$process = $this->getProcessExecutorMock();
$process->expects(array(
$command,
), true);
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
$process,
))
->onlyMethods(array('getListeners'))
->getMock();
$listener = array($command);
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listener));
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
}
2016-07-21 22:20:56 +00:00
/**
* @dataProvider provideDevModes
*
2016-07-21 22:20:56 +00:00
* @param bool $devMode
*/
public function testDispatcherPassDevModeToAutoloadGeneratorForScriptEvents($devMode): void
2016-07-21 22:20:56 +00:00
{
$composer = $this->createComposerInstance();
$generator = $this->getGeneratorMockForDevModePassingTest();
$generator->expects($this->atLeastOnce())
->method('setDevMode')
->with($devMode);
$composer->setAutoloadGenerator($generator);
$package = $this->getMockBuilder('Composer\Package\RootPackageInterface')->getMock();
2016-07-21 22:20:56 +00:00
$package->method('getScripts')->will($this->returnValue(array('scriptName' => array('scriptName'))));
$composer->setPackage($package);
$composer->setRepositoryManager($this->getRepositoryManagerMockForDevModePassingTest());
$composer->setInstallationManager($this->getMockBuilder('Composer\Installer\InstallationManager')->disableOriginalConstructor()->getMock());
2016-07-21 22:20:56 +00:00
$dispatcher = new EventDispatcher(
$composer,
$this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
$this->getProcessExecutorMock()
2016-07-21 22:20:56 +00:00
);
$event = $this->getMockBuilder('Composer\Script\Event')
->disableOriginalConstructor()
->getMock();
$event->method('getName')->will($this->returnValue('scriptName'));
$event->expects($this->atLeastOnce())
->method('isDevMode')
->will($this->returnValue($devMode));
$dispatcher->hasEventListeners($event);
}
2022-02-21 12:42:28 +00:00
public function provideDevModes(): array
2016-07-21 22:20:56 +00:00
{
return array(
array(true),
array(false),
);
}
/**
* @return \PHPUnit\Framework\MockObject\MockObject&\Composer\Autoload\AutoloadGenerator
*/
2016-07-21 22:20:56 +00:00
private function getGeneratorMockForDevModePassingTest()
{
$generator = $this->getMockBuilder('Composer\Autoload\AutoloadGenerator')
->disableOriginalConstructor()
->onlyMethods(array(
2016-07-21 22:20:56 +00:00
'buildPackageMap',
'parseAutoloads',
'createLoader',
'setDevMode',
))
->getMock();
$generator
->method('buildPackageMap')
->will($this->returnValue(array()));
$generator
->method('parseAutoloads')
2022-02-18 07:50:11 +00:00
->will($this->returnValue(array('psr-0' => array(), 'psr-4' => array(), 'classmap' => array(), 'files' => array(), 'exclude-from-classmap' => array())));
2016-07-21 22:20:56 +00:00
$generator
->method('createLoader')
->will($this->returnValue($this->getMockBuilder('Composer\Autoload\ClassLoader')->getMock()));
2016-07-21 22:20:56 +00:00
return $generator;
}
/**
* @return \PHPUnit\Framework\MockObject\MockObject&\Composer\Repository\RepositoryManager
*/
2016-07-21 22:20:56 +00:00
private function getRepositoryManagerMockForDevModePassingTest()
{
$rm = $this->getMockBuilder('Composer\Repository\RepositoryManager')
->disableOriginalConstructor()
->onlyMethods(array('getLocalRepository'))
2016-07-21 22:20:56 +00:00
->getMock();
$repo = $this->getMockBuilder('Composer\Repository\InstalledRepositoryInterface')->getMock();
2016-07-21 22:20:56 +00:00
$repo
->method('getCanonicalPackages')
->will($this->returnValue(array()));
$rm
->method('getLocalRepository')
->will($this->returnValue($repo));
return $rm;
}
public function testDispatcherRemoveListener(): void
{
$composer = $this->createComposerInstance();
$composer->setRepositoryManager($this->getRepositoryManagerMockForDevModePassingTest());
$composer->setInstallationManager($this->getMockBuilder('Composer\Installer\InstallationManager')->disableOriginalConstructor()->getMock());
$dispatcher = new EventDispatcher(
$composer,
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
$this->getProcessExecutorMock()
);
$listener = array($this, 'someMethod');
$listener2 = array($this, 'someMethod2');
$listener3 = 'Composer\\Test\\EventDispatcher\\EventDispatcherTest::someMethod';
$dispatcher->addListener('ev1', $listener, 0);
$dispatcher->addListener('ev1', $listener, 1);
$dispatcher->addListener('ev1', $listener2, 1);
$dispatcher->addListener('ev1', $listener3);
$dispatcher->addListener('ev2', $listener3);
$dispatcher->addListener('ev2', $listener);
$dispatcher->dispatch('ev1');
$dispatcher->dispatch('ev2');
$expected = '> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL
.'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod2'.PHP_EOL
.'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL
.'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
.'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
.'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
$dispatcher->removeListener($this);
$dispatcher->dispatch('ev1');
$dispatcher->dispatch('ev2');
$expected .= '> ev1: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
.'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
}
public function testDispatcherCanExecuteCliAndPhpInSameEventScriptStack(): void
{
$process = $this->getProcessExecutorMock();
$process->expects(array(
'echo -n foo',
'echo -n bar',
), true);
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
$process,
))
->onlyMethods(array(
'getListeners',
))
->getMock();
$listeners = array(
'echo -n foo',
'Composer\\Test\\EventDispatcher\\EventDispatcherTest::someMethod',
'echo -n bar',
);
2015-06-24 07:21:36 +00:00
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listeners));
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
$expected = '> post-install-cmd: echo -n foo'.PHP_EOL.
'> post-install-cmd: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL.
'> post-install-cmd: echo -n bar'.PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
}
public function testDispatcherCanPutEnv(): void
{
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
$this->createComposerInstance(),
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
$this->getProcessExecutorMock(),
))
->onlyMethods(array(
'getListeners',
))
->getMock();
$listeners = array(
'@putenv ABC=123',
'Composer\\Test\\EventDispatcher\\EventDispatcherTest::getTestEnv',
);
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listeners));
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
$expected = '> post-install-cmd: @putenv ABC=123'.PHP_EOL.
'> post-install-cmd: Composer\Test\EventDispatcher\EventDispatcherTest::getTestEnv'.PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
}
public function testDispatcherAppendsDirBinOnPathForEveryListener(): void
{
$currentDirectoryBkp = getcwd();
$composerBinDirBkp = Platform::getEnv('COMPOSER_BIN_DIR');
chdir(__DIR__);
Platform::putEnv('COMPOSER_BIN_DIR', __DIR__ . '/vendor/bin');
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')->setConstructorArgs(array(
$this->createComposerInstance(),
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
$this->getProcessExecutorMock(),
))->onlyMethods(array(
'getListeners',
))->getMock();
$listeners = array(
'Composer\\Test\\EventDispatcher\\EventDispatcherTest::createsVendorBinFolderChecksEnvDoesNotContainsBin',
'Composer\\Test\\EventDispatcher\\EventDispatcherTest::createsVendorBinFolderChecksEnvContainsBin',
);
$dispatcher->expects($this->atLeastOnce())->method('getListeners')->will($this->returnValue($listeners));
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
rmdir(__DIR__ . '/vendor/bin');
rmdir(__DIR__ . '/vendor');
chdir($currentDirectoryBkp);
if ($composerBinDirBkp) {
Platform::putEnv('COMPOSER_BIN_DIR', $composerBinDirBkp);
} else {
Platform::clearEnv('COMPOSER_BIN_DIR');
}
}
/**
* @return void
*/
public static function createsVendorBinFolderChecksEnvDoesNotContainsBin(): void
{
mkdir(__DIR__ . '/vendor/bin', 0700, true);
$val = getenv('PATH');
if (!$val) {
$val = getenv('Path');
}
self::assertStringNotContainsString(__DIR__ . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'bin', $val);
}
/**
* @return void
*/
public static function createsVendorBinFolderChecksEnvContainsBin(): void
{
$val = getenv('PATH');
if (!$val) {
$val = getenv('Path');
}
self::assertStringContainsString(__DIR__ . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'bin', $val);
}
/**
* @return void
*/
public static function getTestEnv(): void
2020-11-22 13:48:56 +00:00
{
$val = getenv('ABC');
if ($val !== '123') {
throw new \Exception('getenv() did not return the expected value. expected 123 got '. var_export($val, true));
}
}
public function testDispatcherCanExecuteComposerScriptGroups(): void
{
$process = $this->getProcessExecutorMock();
$process->expects(array(
'echo -n foo',
'echo -n baz',
'echo -n bar',
), true);
2015-11-21 19:28:10 +00:00
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$composer = $this->createComposerInstance(),
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
$process,
))
->onlyMethods(array(
'getListeners',
))
->getMock();
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
2022-02-21 12:42:28 +00:00
->will($this->returnCallback(function (Event $event): array {
if ($event->getName() === 'root') {
return array('@group');
2016-05-17 10:34:54 +00:00
}
if ($event->getName() === 'group') {
return array('echo -n foo', '@subgroup', 'echo -n bar');
2016-05-17 10:34:54 +00:00
}
if ($event->getName() === 'subgroup') {
return array('echo -n baz');
}
return array();
}));
2018-11-12 14:57:44 +00:00
$dispatcher->dispatch('root', new ScriptEvent('root', $composer, $io));
$expected = '> root: @group'.PHP_EOL.
'> group: echo -n foo'.PHP_EOL.
'> group: @subgroup'.PHP_EOL.
'> subgroup: echo -n baz'.PHP_EOL.
'> group: echo -n bar'.PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
}
public function testRecursionInScriptsNames(): void
{
$process = $this->getProcessExecutorMock();
$process->expects(array(
'echo Hello '.ProcessExecutor::escape('World'),
), true);
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
$composer = $this->createComposerInstance(),
$io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
2020-11-22 13:48:56 +00:00
$process,
))
->onlyMethods(array(
2020-11-22 13:48:56 +00:00
'getListeners',
))
->getMock();
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
2022-02-21 12:42:28 +00:00
->will($this->returnCallback(function (Event $event): array {
2020-11-22 13:48:56 +00:00
if ($event->getName() === 'hello') {
return array('echo Hello');
}
2020-11-22 13:48:56 +00:00
if ($event->getName() === 'helloWorld') {
return array('@hello World');
}
return array();
}));
2018-12-03 09:59:04 +00:00
$dispatcher->dispatch('helloWorld', new ScriptEvent('helloWorld', $composer, $io));
$expected = "> helloWorld: @hello World".PHP_EOL.
2021-10-08 20:46:07 +00:00
"> hello: echo Hello " .$this->getCmd("'World'").PHP_EOL;
$this->assertEquals($expected, $io->getOutput());
}
public function testDispatcherDetectInfiniteRecursion(): void
2015-11-09 12:05:16 +00:00
{
2021-12-09 19:55:26 +00:00
self::expectException('RuntimeException');
2015-11-09 12:05:16 +00:00
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$composer = $this->createComposerInstance(),
$io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
$this->getProcessExecutorMock(),
2015-11-09 12:05:16 +00:00
))
->onlyMethods(array(
2015-11-09 12:05:16 +00:00
'getListeners',
))
->getMock();
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
2022-02-21 12:42:28 +00:00
->will($this->returnCallback(function (Event $event): array {
2015-11-09 12:05:16 +00:00
if ($event->getName() === 'root') {
return array('@recurse');
2016-05-17 10:34:54 +00:00
}
if ($event->getName() === 'recurse') {
2015-11-09 12:05:16 +00:00
return array('@root');
}
return array();
}));
2018-11-12 14:57:44 +00:00
$dispatcher->dispatch('root', new ScriptEvent('root', $composer, $io));
2015-11-09 12:05:16 +00:00
}
/**
* @param array<callable|string> $listeners
*
* @return \PHPUnit\Framework\MockObject\MockObject&\Composer\EventDispatcher\EventDispatcher
*/
private function getDispatcherStubForListenersTest($listeners, IOInterface $io)
{
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$io,
))
->onlyMethods(array('getListeners'))
->getMock();
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listeners));
return $dispatcher;
}
2022-02-21 12:42:28 +00:00
public function provideValidCommands(): array
{
return array(
array('phpunit'),
array('echo foo'),
array('echo -n foo'),
);
}
public function testDispatcherOutputsCommand(): void
2012-12-06 08:56:27 +00:00
{
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
2012-12-06 08:56:27 +00:00
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
new ProcessExecutor($io),
2012-12-06 08:56:27 +00:00
))
->onlyMethods(array('getListeners'))
2012-12-06 08:56:27 +00:00
->getMock();
$listener = array('echo foo');
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listener));
2015-06-09 07:02:32 +00:00
$io->expects($this->once())
->method('writeError')
->with($this->equalTo('> echo foo'));
$io->expects($this->once())
2020-01-14 14:46:58 +00:00
->method('writeRaw')
2016-12-06 15:03:03 +00:00
->with($this->equalTo('foo'.PHP_EOL), false);
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
2012-12-06 08:56:27 +00:00
}
public function testDispatcherOutputsErrorOnFailedCommand(): void
2012-12-06 08:56:27 +00:00
{
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
2012-12-06 08:56:27 +00:00
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
2012-12-06 08:56:27 +00:00
new ProcessExecutor,
))
->onlyMethods(array('getListeners'))
2012-12-06 08:56:27 +00:00
->getMock();
$code = 'exit 1';
2012-12-06 08:56:27 +00:00
$listener = array($code);
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue($listener));
$io->expects($this->once())
2015-07-04 11:22:58 +00:00
->method('isVerbose')
->willReturn(0);
$io->expects($this->atLeast(2))
2015-06-09 11:40:13 +00:00
->method('writeError')
->withConsecutive(
['> exit 1'],
['<error>Script '.$code.' handling the post-install-cmd event returned with error code 1</error>']
);
2015-06-09 11:40:13 +00:00
$io->expects($this->once())
->method('isInteractive')
->willReturn(1);
2021-12-09 19:55:26 +00:00
self::expectException('RuntimeException');
$dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
2012-12-06 08:56:27 +00:00
}
public function testDispatcherInstallerEvents(): void
2014-07-29 13:25:16 +00:00
{
$dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
->setConstructorArgs(array(
2016-02-10 14:35:53 +00:00
$this->createComposerInstance(),
$this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
$this->getProcessExecutorMock(),
2014-07-29 13:25:16 +00:00
))
->onlyMethods(array('getListeners'))
2014-07-29 13:25:16 +00:00
->getMock();
$dispatcher->expects($this->atLeastOnce())
->method('getListeners')
->will($this->returnValue(array()));
$transaction = $this->getMockBuilder('Composer\DependencyResolver\LockTransaction')->disableOriginalConstructor()->getMock();
2014-07-29 13:25:16 +00:00
$dispatcher->dispatchInstallerEvent(InstallerEvents::PRE_OPERATIONS_EXEC, true, true, $transaction);
2014-07-29 13:25:16 +00:00
}
/**
* @return void
*/
public static function call(): void
{
throw new \RuntimeException();
}
/**
* @return true
*/
public static function someMethod(): bool
{
return true;
}
2016-02-10 14:35:53 +00:00
/**
* @return true
*/
public static function someMethod2(): bool
{
return true;
}
/**
* @return Composer
*/
private function createComposerInstance(): Composer
2016-02-10 14:35:53 +00:00
{
$composer = new Composer;
$config = new Config();
2016-02-10 14:35:53 +00:00
$composer->setConfig($config);
$package = $this->getMockBuilder('Composer\Package\RootPackageInterface')->getMock();
$composer->setPackage($package);
2016-02-10 14:35:53 +00:00
return $composer;
}
2012-06-14 10:10:01 +00:00
}