loadPlugins(['TestPlugin', 'Company/TestPluginThree']); static::setAppNamespace(); $this->dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['_stop']) ->getMock(); } /** * teardown * * @return void */ public function tearDown() { parent::tearDown(); ShellDispatcher::resetAliases(); $this->clearPlugins(); } /** * Test error on missing shell * * @return void */ public function testFindShellMissing() { $this->expectException(\Cake\Console\Exception\MissingShellException::class); $this->dispatcher->findShell('nope'); } /** * Test error on missing plugin shell * * @return void */ public function testFindShellMissingPlugin() { $this->expectException(\Cake\Console\Exception\MissingShellException::class); $this->dispatcher->findShell('test_plugin.nope'); } /** * Verify loading of (plugin-) shells * * @return void */ public function testFindShell() { $result = $this->dispatcher->findShell('sample'); $this->assertInstanceOf('TestApp\Shell\SampleShell', $result); $result = $this->dispatcher->findShell('test_plugin.example'); $this->assertInstanceOf('TestPlugin\Shell\ExampleShell', $result); $this->assertEquals('TestPlugin', $result->plugin); $this->assertEquals('Example', $result->name); $result = $this->dispatcher->findShell('TestPlugin.example'); $this->assertInstanceOf('TestPlugin\Shell\ExampleShell', $result); $result = $this->dispatcher->findShell('TestPlugin.Example'); $this->assertInstanceOf('TestPlugin\Shell\ExampleShell', $result); } /** * testAddShortPluginAlias * * @return void */ public function testAddShortPluginAlias() { $expected = [ 'Company' => 'Company/TestPluginThree.company', 'Example' => 'TestPlugin.example', ]; $result = $this->dispatcher->addShortPluginAliases(); $this->assertSame($expected, $result, 'Should return the list of aliased plugin shells'); ShellDispatcher::alias('Example', 'SomeOther.PluginsShell'); $expected = [ 'Company' => 'Company/TestPluginThree.company', 'Example' => 'SomeOther.PluginsShell', ]; $result = $this->dispatcher->addShortPluginAliases(); $this->assertSame($expected, $result, 'Should not overwrite existing aliases'); } /** * Test getting shells with aliases. * * @return void */ public function testFindShellAliased() { ShellDispatcher::alias('short', 'test_plugin.example'); $result = $this->dispatcher->findShell('short'); $this->assertInstanceOf('TestPlugin\Shell\ExampleShell', $result); $this->assertEquals('TestPlugin', $result->plugin); $this->assertEquals('Example', $result->name); } /** * Test finding a shell that has a matching alias. * * Aliases should not overload concrete shells. * * @return void */ public function testFindShellAliasedAppShadow() { ShellDispatcher::alias('sample', 'test_plugin.example'); $result = $this->dispatcher->findShell('sample'); $this->assertInstanceOf('TestApp\Shell\SampleShell', $result); $this->assertEmpty($result->plugin); $this->assertEquals('Sample', $result->name); } /** * Verify dispatch handling stop errors * * @return void */ public function testDispatchShellWithAbort() { $io = $this->getMockBuilder('Cake\Console\ConsoleIo')->getMock(); $shell = $this->getMockBuilder('Cake\Console\Shell') ->setMethods(['main']) ->setConstructorArgs([$io]) ->getMock(); $shell->expects($this->once()) ->method('main') ->will($this->returnCallback(function () use ($shell) { $shell->abort('Bad things', 99); })); $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell']) ->getMock(); $dispatcher->expects($this->any()) ->method('findShell') ->with('aborter') ->will($this->returnValue($shell)); $dispatcher->args = ['aborter']; $result = $dispatcher->dispatch(); $this->assertSame(99, $result, 'Should return the exception error code.'); } /** * Verify correct dispatch of Shell subclasses with a main method * * @return void */ public function testDispatchShellWithMain() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $Shell->expects($this->exactly(2))->method('initialize'); $Shell->expects($this->at(0))->method('runCommand') ->will($this->returnValue(true)); $Shell->expects($this->at(1))->method('runCommand') ->will($this->returnValue(null)); $dispatcher->expects($this->any()) ->method('findShell') ->with('mock_with_main') ->will($this->returnValue($Shell)); $dispatcher->args = ['mock_with_main']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); $this->assertEquals([], $dispatcher->args); $dispatcher->args = ['mock_with_main']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); $this->assertEquals([], $dispatcher->args); } /** * Verifies correct dispatch of Shell subclasses with integer exit codes. * * @return void */ public function testDispatchShellWithIntegerSuccessCode() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $Shell->expects($this->once())->method('initialize'); $Shell->expects($this->once())->method('runCommand') ->with(['initdb']) ->will($this->returnValue(Shell::CODE_SUCCESS)); $dispatcher->expects($this->any()) ->method('findShell') ->with('mock_without_main') ->will($this->returnValue($Shell)); $dispatcher->args = ['mock_without_main', 'initdb']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); } /** * Verifies correct dispatch of Shell subclasses with custom integer exit codes. * * @return void */ public function testDispatchShellWithCustomIntegerCodes() { $customErrorCode = 3; $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $Shell->expects($this->once())->method('initialize'); $Shell->expects($this->once())->method('runCommand') ->with(['initdb']) ->will($this->returnValue($customErrorCode)); $dispatcher->expects($this->any()) ->method('findShell') ->with('mock_without_main') ->will($this->returnValue($Shell)); $dispatcher->args = ['mock_without_main', 'initdb']; $result = $dispatcher->dispatch(); $this->assertSame($customErrorCode, $result); } /** * Verify correct dispatch of Shell subclasses without a main method * * @return void */ public function testDispatchShellWithoutMain() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $Shell->expects($this->once())->method('initialize'); $Shell->expects($this->once())->method('runCommand') ->with(['initdb']) ->will($this->returnValue(true)); $dispatcher->expects($this->any()) ->method('findShell') ->with('mock_without_main') ->will($this->returnValue($Shell)); $dispatcher->args = ['mock_without_main', 'initdb']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); } /** * Verify you can dispatch a plugin's main shell with the shell name alone * * @return void */ public function testDispatchShortPluginAlias() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['_shellExists', '_createShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $dispatcher->expects($this->at(1)) ->method('_shellExists') ->with('TestPlugin.Example') ->will($this->returnValue('TestPlugin\Console\Command\TestPluginShell')); $dispatcher->expects($this->at(2)) ->method('_createShell') ->with('TestPlugin\Console\Command\TestPluginShell', 'TestPlugin.Example') ->will($this->returnValue($Shell)); $dispatcher->args = ['example']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); } /** * Ensure short plugin shell usage is case/camelized insensitive * * @return void */ public function testDispatchShortPluginAliasCamelized() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['_shellExists', '_createShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $dispatcher->expects($this->at(1)) ->method('_shellExists') ->with('TestPlugin.Example') ->will($this->returnValue('TestPlugin\Console\Command\TestPluginShell')); $dispatcher->expects($this->at(2)) ->method('_createShell') ->with('TestPlugin\Console\Command\TestPluginShell', 'TestPlugin.Example') ->will($this->returnValue($Shell)); $dispatcher->args = ['Example']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); } /** * Verify that in case of conflict, app shells take precedence in alias list * * @return void */ public function testDispatchShortPluginAliasConflict() { $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['_shellExists', '_createShell']) ->getMock(); $Shell = $this->getMockBuilder('Cake\Console\Shell') ->disableOriginalConstructor() ->getMock(); $dispatcher->expects($this->at(1)) ->method('_shellExists') ->with('Sample') ->will($this->returnValue('App\Shell\SampleShell')); $dispatcher->expects($this->at(2)) ->method('_createShell') ->with('App\Shell\SampleShell', 'Sample') ->will($this->returnValue($Shell)); $dispatcher->args = ['sample']; $result = $dispatcher->dispatch(); $this->assertSame(Shell::CODE_SUCCESS, $result); } /** * Verify shifting of arguments * * @return void */ public function testShiftArgs() { $this->dispatcher->args = ['a', 'b', 'c']; $this->assertEquals('a', $this->dispatcher->shiftArgs()); $this->assertSame($this->dispatcher->args, ['b', 'c']); $this->dispatcher->args = ['a' => 'b', 'c', 'd']; $this->assertEquals('b', $this->dispatcher->shiftArgs()); $this->assertSame($this->dispatcher->args, ['c', 'd']); $this->dispatcher->args = ['a', 'b' => 'c', 'd']; $this->assertEquals('a', $this->dispatcher->shiftArgs()); $this->assertSame($this->dispatcher->args, ['b' => 'c', 'd']); $this->dispatcher->args = [0 => 'a', 2 => 'b', 30 => 'c']; $this->assertEquals('a', $this->dispatcher->shiftArgs()); $this->assertSame($this->dispatcher->args, [0 => 'b', 1 => 'c']); $this->dispatcher->args = []; $this->assertNull($this->dispatcher->shiftArgs()); $this->assertSame([], $this->dispatcher->args); } /** * Test how `bin/cake --help` works. * * @return void */ public function testHelpOption() { $mockShell = $this->getMockBuilder('Cake\Shell\CommandListShell') ->setMethods(['main', 'initialize', 'startup']) ->getMock(); $mockShell->expects($this->once()) ->method('main'); $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell', '_stop']) ->getMock(); $dispatcher->expects($this->once()) ->method('findShell') ->with('command_list') ->will($this->returnValue($mockShell)); $dispatcher->args = ['--help']; $dispatcher->dispatch(); } /** * Test how `bin/cake --version` works. * * @return void */ public function testVersionOption() { $mockShell = $this->getMockBuilder('Cake\Shell\CommandListShell') ->setMethods(['main', 'initialize', 'startup']) ->getMock(); $mockShell->expects($this->once()) ->method('main'); $dispatcher = $this->getMockBuilder('Cake\Console\ShellDispatcher') ->setMethods(['findShell', '_stop']) ->getMock(); $dispatcher->expects($this->once()) ->method('findShell') ->with('command_list') ->will($this->returnValue($mockShell)); $dispatcher->args = ['--version']; $dispatcher->dispatch(); } }