EventDispatcherTest.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  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\EventDispatcher;
  12. use Composer\EventDispatcher\Event;
  13. use Composer\EventDispatcher\EventDispatcher;
  14. use Composer\EventDispatcher\ScriptExecutionException;
  15. use Composer\Installer\InstallerEvents;
  16. use Composer\Config;
  17. use Composer\Composer;
  18. use Composer\Test\TestCase;
  19. use Composer\IO\BufferIO;
  20. use Composer\Script\ScriptEvents;
  21. use Composer\Script\Event as ScriptEvent;
  22. use Composer\Util\ProcessExecutor;
  23. use Symfony\Component\Console\Output\OutputInterface;
  24. class EventDispatcherTest extends TestCase
  25. {
  26. /**
  27. * @expectedException RuntimeException
  28. */
  29. public function testListenerExceptionsAreCaught()
  30. {
  31. $io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock();
  32. $dispatcher = $this->getDispatcherStubForListenersTest(array(
  33. 'Composer\Test\EventDispatcher\EventDispatcherTest::call',
  34. ), $io);
  35. $io->expects($this->at(0))
  36. ->method('isVerbose')
  37. ->willReturn(0);
  38. $io->expects($this->at(1))
  39. ->method('writeError')
  40. ->with('> Composer\Test\EventDispatcher\EventDispatcherTest::call');
  41. $io->expects($this->at(2))
  42. ->method('writeError')
  43. ->with('<error>Script Composer\Test\EventDispatcher\EventDispatcherTest::call handling the post-install-cmd event terminated with an exception</error>');
  44. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  45. }
  46. /**
  47. * @dataProvider getValidCommands
  48. * @param string $command
  49. */
  50. public function testDispatcherCanExecuteSingleCommandLineScript($command)
  51. {
  52. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  53. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  54. ->setConstructorArgs(array(
  55. $this->createComposerInstance(),
  56. $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  57. $process,
  58. ))
  59. ->setMethods(array('getListeners'))
  60. ->getMock();
  61. $listener = array($command);
  62. $dispatcher->expects($this->atLeastOnce())
  63. ->method('getListeners')
  64. ->will($this->returnValue($listener));
  65. $process->expects($this->once())
  66. ->method('execute')
  67. ->with($command)
  68. ->will($this->returnValue(0));
  69. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  70. }
  71. /**
  72. * @dataProvider getDevModes
  73. * @param bool $devMode
  74. */
  75. public function testDispatcherPassDevModeToAutoloadGeneratorForScriptEvents($devMode)
  76. {
  77. $composer = $this->createComposerInstance();
  78. $generator = $this->getGeneratorMockForDevModePassingTest();
  79. $generator->expects($this->atLeastOnce())
  80. ->method('setDevMode')
  81. ->with($devMode);
  82. $composer->setAutoloadGenerator($generator);
  83. $package = $this->getMockBuilder('Composer\Package\RootPackageInterface')->getMock();
  84. $package->method('getScripts')->will($this->returnValue(array('scriptName' => array('scriptName'))));
  85. $composer->setPackage($package);
  86. $composer->setRepositoryManager($this->getRepositoryManagerMockForDevModePassingTest());
  87. $composer->setInstallationManager($this->getMockBuilder('Composer\Installer\InstallationManager')->disableOriginalConstructor()->getMock());
  88. $dispatcher = new EventDispatcher(
  89. $composer,
  90. $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  91. $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock()
  92. );
  93. $event = $this->getMockBuilder('Composer\Script\Event')
  94. ->disableOriginalConstructor()
  95. ->getMock();
  96. $event->method('getName')->will($this->returnValue('scriptName'));
  97. $event->expects($this->atLeastOnce())
  98. ->method('isDevMode')
  99. ->will($this->returnValue($devMode));
  100. $dispatcher->hasEventListeners($event);
  101. }
  102. public function getDevModes()
  103. {
  104. return array(
  105. array(true),
  106. array(false),
  107. );
  108. }
  109. private function getGeneratorMockForDevModePassingTest()
  110. {
  111. $generator = $this->getMockBuilder('Composer\Autoload\AutoloadGenerator')
  112. ->disableOriginalConstructor()
  113. ->setMethods(array(
  114. 'buildPackageMap',
  115. 'parseAutoloads',
  116. 'createLoader',
  117. 'setDevMode',
  118. ))
  119. ->getMock();
  120. $generator
  121. ->method('buildPackageMap')
  122. ->will($this->returnValue(array()));
  123. $generator
  124. ->method('parseAutoloads')
  125. ->will($this->returnValue(array()));
  126. $generator
  127. ->method('createLoader')
  128. ->will($this->returnValue($this->getMockBuilder('Composer\Autoload\ClassLoader')->getMock()));
  129. return $generator;
  130. }
  131. private function getRepositoryManagerMockForDevModePassingTest()
  132. {
  133. $rm = $this->getMockBuilder('Composer\Repository\RepositoryManager')
  134. ->disableOriginalConstructor()
  135. ->setMethods(array('getLocalRepository'))
  136. ->getMock();
  137. $repo = $this->getMockBuilder('Composer\Repository\InstalledRepositoryInterface')->getMock();
  138. $repo
  139. ->method('getCanonicalPackages')
  140. ->will($this->returnValue(array()));
  141. $rm
  142. ->method('getLocalRepository')
  143. ->will($this->returnValue($repo));
  144. return $rm;
  145. }
  146. public function testDispatcherRemoveListener()
  147. {
  148. $composer = $this->createComposerInstance();
  149. $composer->setRepositoryManager($this->getRepositoryManagerMockForDevModePassingTest());
  150. $composer->setInstallationManager($this->getMockBuilder('Composer\Installer\InstallationManager')->disableOriginalConstructor()->getMock());
  151. $dispatcher = new EventDispatcher(
  152. $composer,
  153. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
  154. $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock()
  155. );
  156. $listener = array($this, 'someMethod');
  157. $listener2 = array($this, 'someMethod2');
  158. $listener3 = 'Composer\\Test\\EventDispatcher\\EventDispatcherTest::someMethod';
  159. $dispatcher->addListener('ev1', $listener, 0);
  160. $dispatcher->addListener('ev1', $listener, 1);
  161. $dispatcher->addListener('ev1', $listener2, 1);
  162. $dispatcher->addListener('ev1', $listener3);
  163. $dispatcher->addListener('ev2', $listener3);
  164. $dispatcher->addListener('ev2', $listener);
  165. $dispatcher->dispatch('ev1');
  166. $dispatcher->dispatch('ev2');
  167. $expected = '> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL
  168. .'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod2'.PHP_EOL
  169. .'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL
  170. .'> ev1: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
  171. .'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
  172. .'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest->someMethod'.PHP_EOL;
  173. $this->assertEquals($expected, $io->getOutput());
  174. $dispatcher->removeListener($this);
  175. $dispatcher->dispatch('ev1');
  176. $dispatcher->dispatch('ev2');
  177. $expected .= '> ev1: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL
  178. .'> ev2: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL;
  179. $this->assertEquals($expected, $io->getOutput());
  180. }
  181. public function testDispatcherCanExecuteCliAndPhpInSameEventScriptStack()
  182. {
  183. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  184. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  185. ->setConstructorArgs(array(
  186. $this->createComposerInstance(),
  187. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
  188. $process,
  189. ))
  190. ->setMethods(array(
  191. 'getListeners',
  192. ))
  193. ->getMock();
  194. $process->expects($this->exactly(2))
  195. ->method('execute')
  196. ->will($this->returnValue(0));
  197. $listeners = array(
  198. 'echo -n foo',
  199. 'Composer\\Test\\EventDispatcher\\EventDispatcherTest::someMethod',
  200. 'echo -n bar',
  201. );
  202. $dispatcher->expects($this->atLeastOnce())
  203. ->method('getListeners')
  204. ->will($this->returnValue($listeners));
  205. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  206. $expected = '> post-install-cmd: echo -n foo'.PHP_EOL.
  207. '> post-install-cmd: Composer\Test\EventDispatcher\EventDispatcherTest::someMethod'.PHP_EOL.
  208. '> post-install-cmd: echo -n bar'.PHP_EOL;
  209. $this->assertEquals($expected, $io->getOutput());
  210. }
  211. public function testDispatcherCanPutEnv()
  212. {
  213. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  214. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  215. ->setConstructorArgs(array(
  216. $this->createComposerInstance(),
  217. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
  218. $process,
  219. ))
  220. ->setMethods(array(
  221. 'getListeners',
  222. ))
  223. ->getMock();
  224. $listeners = array(
  225. '@putenv ABC=123',
  226. 'Composer\\Test\\EventDispatcher\\EventDispatcherTest::getTestEnv',
  227. );
  228. $dispatcher->expects($this->atLeastOnce())
  229. ->method('getListeners')
  230. ->will($this->returnValue($listeners));
  231. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  232. $expected = '> post-install-cmd: @putenv ABC=123'.PHP_EOL.
  233. '> post-install-cmd: Composer\Test\EventDispatcher\EventDispatcherTest::getTestEnv'.PHP_EOL;
  234. $this->assertEquals($expected, $io->getOutput());
  235. }
  236. static public function getTestEnv() {
  237. $val = getenv('ABC');
  238. if ($val !== '123') {
  239. throw new \Exception('getenv() did not return the expected value. expected 123 got '. var_export($val, true));
  240. }
  241. }
  242. public function testDispatcherCanExecuteComposerScriptGroups()
  243. {
  244. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  245. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  246. ->setConstructorArgs(array(
  247. $composer = $this->createComposerInstance(),
  248. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
  249. $process,
  250. ))
  251. ->setMethods(array(
  252. 'getListeners',
  253. ))
  254. ->getMock();
  255. $process->expects($this->exactly(3))
  256. ->method('execute')
  257. ->will($this->returnValue(0));
  258. $dispatcher->expects($this->atLeastOnce())
  259. ->method('getListeners')
  260. ->will($this->returnCallback(function (Event $event) {
  261. if ($event->getName() === 'root') {
  262. return array('@group');
  263. }
  264. if ($event->getName() === 'group') {
  265. return array('echo -n foo', '@subgroup', 'echo -n bar');
  266. }
  267. if ($event->getName() === 'subgroup') {
  268. return array('echo -n baz');
  269. }
  270. return array();
  271. }));
  272. $dispatcher->dispatch('root', new ScriptEvent('root', $composer, $io));
  273. $expected = '> root: @group'.PHP_EOL.
  274. '> group: echo -n foo'.PHP_EOL.
  275. '> group: @subgroup'.PHP_EOL.
  276. '> subgroup: echo -n baz'.PHP_EOL.
  277. '> group: echo -n bar'.PHP_EOL;
  278. $this->assertEquals($expected, $io->getOutput());
  279. }
  280. public function testRecursionInScriptsNames()
  281. {
  282. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  283. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  284. ->setConstructorArgs(array(
  285. $composer = $this->createComposerInstance(),
  286. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE),
  287. $process
  288. ))
  289. ->setMethods(array(
  290. 'getListeners'
  291. ))
  292. ->getMock();
  293. $process->expects($this->exactly(1))
  294. ->method('execute')
  295. ->will($this->returnValue(0));
  296. $dispatcher->expects($this->atLeastOnce())
  297. ->method('getListeners')
  298. ->will($this->returnCallback(function (Event $event) {
  299. if($event->getName() === 'hello') {
  300. return array('echo Hello');
  301. }
  302. if($event->getName() === 'helloWorld') {
  303. return array('@hello World');
  304. }
  305. return array();
  306. }));
  307. $dispatcher->dispatch('helloWorld', new ScriptEvent('helloWorld', $composer, $io));
  308. $expected = "> helloWorld: @hello World".PHP_EOL.
  309. "> hello: echo Hello " .escapeshellarg('World').PHP_EOL;
  310. $this->assertEquals($expected, $io->getOutput());
  311. }
  312. /**
  313. * @expectedException RuntimeException
  314. */
  315. public function testDispatcherDetectInfiniteRecursion()
  316. {
  317. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  318. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  319. ->setConstructorArgs(array(
  320. $composer = $this->createComposerInstance(),
  321. $io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  322. $process,
  323. ))
  324. ->setMethods(array(
  325. 'getListeners',
  326. ))
  327. ->getMock();
  328. $dispatcher->expects($this->atLeastOnce())
  329. ->method('getListeners')
  330. ->will($this->returnCallback(function (Event $event) {
  331. if ($event->getName() === 'root') {
  332. return array('@recurse');
  333. }
  334. if ($event->getName() === 'recurse') {
  335. return array('@root');
  336. }
  337. return array();
  338. }));
  339. $dispatcher->dispatch('root', new ScriptEvent('root', $composer, $io));
  340. }
  341. private function getDispatcherStubForListenersTest($listeners, $io)
  342. {
  343. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  344. ->setConstructorArgs(array(
  345. $this->createComposerInstance(),
  346. $io,
  347. ))
  348. ->setMethods(array('getListeners'))
  349. ->getMock();
  350. $dispatcher->expects($this->atLeastOnce())
  351. ->method('getListeners')
  352. ->will($this->returnValue($listeners));
  353. return $dispatcher;
  354. }
  355. public function getValidCommands()
  356. {
  357. return array(
  358. array('phpunit'),
  359. array('echo foo'),
  360. array('echo -n foo'),
  361. );
  362. }
  363. public function testDispatcherOutputsCommand()
  364. {
  365. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  366. ->setConstructorArgs(array(
  367. $this->createComposerInstance(),
  368. $io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  369. new ProcessExecutor($io),
  370. ))
  371. ->setMethods(array('getListeners'))
  372. ->getMock();
  373. $listener = array('echo foo');
  374. $dispatcher->expects($this->atLeastOnce())
  375. ->method('getListeners')
  376. ->will($this->returnValue($listener));
  377. $io->expects($this->once())
  378. ->method('writeError')
  379. ->with($this->equalTo('> echo foo'));
  380. $io->expects($this->once())
  381. ->method('write')
  382. ->with($this->equalTo('foo'.PHP_EOL), false);
  383. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  384. }
  385. public function testDispatcherOutputsErrorOnFailedCommand()
  386. {
  387. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  388. ->setConstructorArgs(array(
  389. $this->createComposerInstance(),
  390. $io = $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  391. new ProcessExecutor,
  392. ))
  393. ->setMethods(array('getListeners'))
  394. ->getMock();
  395. $code = 'exit 1';
  396. $listener = array($code);
  397. $dispatcher->expects($this->atLeastOnce())
  398. ->method('getListeners')
  399. ->will($this->returnValue($listener));
  400. $io->expects($this->at(0))
  401. ->method('isVerbose')
  402. ->willReturn(0);
  403. $io->expects($this->at(1))
  404. ->method('writeError')
  405. ->willReturn('> exit 1');
  406. $io->expects($this->at(2))
  407. ->method('writeError')
  408. ->with($this->equalTo('<error>Script '.$code.' handling the post-install-cmd event returned with error code 1</error>'));
  409. $this->setExpectedException('RuntimeException');
  410. $dispatcher->dispatchScript(ScriptEvents::POST_INSTALL_CMD, false);
  411. }
  412. public function testDispatcherInstallerEvents()
  413. {
  414. $process = $this->getMockBuilder('Composer\Util\ProcessExecutor')->getMock();
  415. $dispatcher = $this->getMockBuilder('Composer\EventDispatcher\EventDispatcher')
  416. ->setConstructorArgs(array(
  417. $this->createComposerInstance(),
  418. $this->getMockBuilder('Composer\IO\IOInterface')->getMock(),
  419. $process,
  420. ))
  421. ->setMethods(array('getListeners'))
  422. ->getMock();
  423. $dispatcher->expects($this->atLeastOnce())
  424. ->method('getListeners')
  425. ->will($this->returnValue(array()));
  426. $policy = $this->getMockBuilder('Composer\DependencyResolver\PolicyInterface')->getMock();
  427. $repositorySet = $this->getMockBuilder('Composer\Repository\RepositorySet')->disableOriginalConstructor()->getMock();
  428. $installedRepo = $this->getMockBuilder('Composer\Repository\CompositeRepository')->disableOriginalConstructor()->getMock();
  429. $request = $this->getMockBuilder('Composer\DependencyResolver\Request')->disableOriginalConstructor()->getMock();
  430. $dispatcher->dispatchInstallerEvent(InstallerEvents::PRE_DEPENDENCIES_SOLVING, true, $policy, $repositorySet, $installedRepo, $request);
  431. $dispatcher->dispatchInstallerEvent(InstallerEvents::POST_DEPENDENCIES_SOLVING, true, $policy, $repositorySet, $installedRepo, $request, array());
  432. }
  433. public static function call()
  434. {
  435. throw new \RuntimeException();
  436. }
  437. public static function someMethod()
  438. {
  439. return true;
  440. }
  441. public static function someMethod2()
  442. {
  443. return true;
  444. }
  445. private function createComposerInstance()
  446. {
  447. $composer = new Composer;
  448. $config = new Config;
  449. $composer->setConfig($config);
  450. $package = $this->getMockBuilder('Composer\Package\RootPackageInterface')->getMock();
  451. $composer->setPackage($package);
  452. return $composer;
  453. }
  454. }