mirror of
https://github.com/yiisoft/yii2.git
synced 2025-08-14 14:28:27 +08:00
301 lines
12 KiB
PHP
301 lines
12 KiB
PHP
<?php
|
|
/**
|
|
* @link https://www.yiiframework.com/
|
|
* @copyright Copyright (c) 2008 Yii Software LLC
|
|
* @license https://www.yiiframework.com/license/
|
|
*/
|
|
|
|
namespace yiiunit\framework\console;
|
|
|
|
use RuntimeException;
|
|
use Yii;
|
|
use yii\base\InlineAction;
|
|
use yii\base\Module;
|
|
use yii\console\Application;
|
|
use yii\console\Exception;
|
|
use yii\console\Request;
|
|
use yii\console\Response;
|
|
use yii\helpers\Console;
|
|
use yiiunit\framework\console\stubs\DummyService;
|
|
use yiiunit\TestCase;
|
|
|
|
/**
|
|
* @group console
|
|
*/
|
|
class ControllerTest extends TestCase
|
|
{
|
|
private FakePhp71Controller|null $controller = null;
|
|
|
|
protected function setUp(): void
|
|
{
|
|
parent::setUp();
|
|
$this->mockApplication();
|
|
Yii::$app->controllerMap = [
|
|
'fake' => 'yiiunit\framework\console\FakeController',
|
|
'fake_witout_output' => 'yiiunit\framework\console\FakeHelpControllerWithoutOutput',
|
|
'help' => 'yiiunit\framework\console\FakeHelpController',
|
|
];
|
|
}
|
|
|
|
public function testBindArrayToActionParams(): void
|
|
{
|
|
$controller = new FakeController('fake', Yii::$app);
|
|
|
|
$params = ['test' => []];
|
|
$this->assertEquals([], $controller->runAction('aksi4', $params));
|
|
$this->assertEquals([], $controller->runAction('aksi4', $params));
|
|
}
|
|
|
|
public function testBindActionParams(): void
|
|
{
|
|
$controller = new FakeController('fake', Yii::$app);
|
|
|
|
$params = ['from params'];
|
|
[$fromParam, $other] = $controller->run('aksi1', $params);
|
|
$this->assertEquals('from params', $fromParam);
|
|
$this->assertEquals('default', $other);
|
|
|
|
$params = ['from params', 'notdefault'];
|
|
[$fromParam, $other] = $controller->run('aksi1', $params);
|
|
$this->assertEquals('from params', $fromParam);
|
|
$this->assertEquals('notdefault', $other);
|
|
|
|
$params = ['d426,mdmunir', 'single'];
|
|
$result = $controller->runAction('aksi2', $params);
|
|
$this->assertEquals([['d426', 'mdmunir'], 'single'], $result);
|
|
|
|
$params = ['', 'single'];
|
|
$result = $controller->runAction('aksi2', $params);
|
|
$this->assertEquals([[], 'single'], $result);
|
|
|
|
$params = ['_aliases' => ['t' => 'test']];
|
|
$result = $controller->runAction('aksi4', $params);
|
|
$this->assertEquals('test', $result);
|
|
|
|
$params = ['_aliases' => ['a' => 'testAlias']];
|
|
$result = $controller->runAction('aksi5', $params);
|
|
$this->assertEquals('testAlias', $result);
|
|
|
|
$params = ['_aliases' => ['ta' => 'from params,notdefault']];
|
|
[$fromParam, $other] = $controller->runAction('aksi6', $params);
|
|
$this->assertEquals('from params', $fromParam);
|
|
$this->assertEquals('notdefault', $other);
|
|
|
|
$params = ['test-array' => 'from params,notdefault'];
|
|
[$fromParam, $other] = $controller->runAction('aksi6', $params);
|
|
$this->assertEquals('from params', $fromParam);
|
|
$this->assertEquals('notdefault', $other);
|
|
|
|
$params = ['from params', 'notdefault'];
|
|
[$fromParam, $other] = $controller->run('trimargs', $params);
|
|
$this->assertEquals('from params', $fromParam);
|
|
$this->assertEquals('notdefault', $other);
|
|
|
|
$params = ['a', 'b', 'c1', 'c2', 'c3'];
|
|
[$a, $b, $c] = $controller->run('variadic', $params);
|
|
$this->assertEquals('a', $a);
|
|
$this->assertEquals('b', $b);
|
|
$this->assertEquals(['c1', 'c2', 'c3'], $c);
|
|
|
|
$params = ['avaliable'];
|
|
$message = Yii::t('yii', 'Missing required arguments: {params}', ['params' => implode(', ', ['missing'])]);
|
|
$this->expectException(Exception::class);
|
|
$this->expectExceptionMessage($message);
|
|
$result = $controller->runAction('aksi3', $params);
|
|
}
|
|
|
|
public function testNullableInjectedActionParams(): void
|
|
{
|
|
// Use the PHP71 controller for this test
|
|
$this->controller = new FakePhp71Controller('fake', new Application([
|
|
'id' => 'app',
|
|
'basePath' => __DIR__,
|
|
]));
|
|
$this->mockApplication(['controller' => $this->controller]);
|
|
|
|
$injectionAction = new InlineAction('injection', $this->controller, 'actionNullableInjection');
|
|
$params = [];
|
|
$args = $this->controller->bindActionParams($injectionAction, $params);
|
|
$this->assertEquals(\Yii::$app->request, $args[0]);
|
|
$this->assertNull($args[1]);
|
|
}
|
|
|
|
public function testInjectionContainerException(): void
|
|
{
|
|
// Use the PHP71 controller for this test
|
|
$this->controller = new FakePhp71Controller('fake', new Application([
|
|
'id' => 'app',
|
|
'basePath' => __DIR__,
|
|
]));
|
|
$this->mockApplication(['controller' => $this->controller]);
|
|
|
|
$injectionAction = new InlineAction('injection', $this->controller, 'actionInjection');
|
|
$params = ['between' => 'test', 'after' => 'another', 'before' => 'test'];
|
|
\Yii::$container->set(DummyService::class, function(): never { throw new \RuntimeException('uh oh'); });
|
|
|
|
$this->expectException(RuntimeException::class);
|
|
$this->expectExceptionMessage('uh oh');
|
|
$this->controller->bindActionParams($injectionAction, $params);
|
|
}
|
|
|
|
public function testUnknownInjection(): void
|
|
{
|
|
// Use the PHP71 controller for this test
|
|
$this->controller = new FakePhp71Controller('fake', new Application([
|
|
'id' => 'app',
|
|
'basePath' => __DIR__,
|
|
]));
|
|
$this->mockApplication(['controller' => $this->controller]);
|
|
|
|
$injectionAction = new InlineAction('injection', $this->controller, 'actionInjection');
|
|
$params = ['between' => 'test', 'after' => 'another', 'before' => 'test'];
|
|
\Yii::$container->clear(DummyService::class);
|
|
$this->expectException(Exception::class);
|
|
$this->expectExceptionMessage('Could not load required service: dummyService');
|
|
$this->controller->bindActionParams($injectionAction, $params);
|
|
}
|
|
|
|
public function testInjectedActionParams(): void
|
|
{
|
|
// Use the PHP71 controller for this test
|
|
$this->controller = new FakePhp71Controller('fake', new Application([
|
|
'id' => 'app',
|
|
'basePath' => __DIR__,
|
|
]));
|
|
$this->mockApplication(['controller' => $this->controller]);
|
|
|
|
$injectionAction = new InlineAction('injection', $this->controller, 'actionInjection');
|
|
$params = ['between' => 'test', 'after' => 'another', 'before' => 'test'];
|
|
\Yii::$container->set(DummyService::class, DummyService::class);
|
|
$args = $this->controller->bindActionParams($injectionAction, $params);
|
|
$this->assertEquals($params['before'], $args[0]);
|
|
$this->assertEquals(\Yii::$app->request, $args[1]);
|
|
$this->assertEquals('Component: yii\console\Request $request', \Yii::$app->requestedParams['request']);
|
|
$this->assertEquals($params['between'], $args[2]);
|
|
$this->assertInstanceOf(DummyService::class, $args[3]);
|
|
$this->assertEquals('Container DI: yiiunit\framework\console\stubs\DummyService $dummyService', \Yii::$app->requestedParams['dummyService']);
|
|
$this->assertNull($args[4]);
|
|
$this->assertEquals('Unavailable service: post', \Yii::$app->requestedParams['post']);
|
|
$this->assertEquals($params['after'], $args[5]);
|
|
}
|
|
|
|
public function testInjectedActionParamsFromModule(): void
|
|
{
|
|
$module = new \yii\base\Module('fake', new Application([
|
|
'id' => 'app',
|
|
'basePath' => __DIR__,
|
|
]));
|
|
$module->set('yii\data\DataProviderInterface', [
|
|
'class' => \yii\data\ArrayDataProvider::class,
|
|
]);
|
|
// Use the PHP71 controller for this test
|
|
$this->controller = new FakePhp71Controller('fake', $module);
|
|
$this->mockWebApplication(['controller' => $this->controller]);
|
|
|
|
$injectionAction = new InlineAction('injection', $this->controller, 'actionModuleServiceInjection');
|
|
$args = $this->controller->bindActionParams($injectionAction, []);
|
|
$this->assertInstanceOf(\yii\data\ArrayDataProvider::class, $args[0]);
|
|
$this->assertEquals('Module yii\base\Module DI: yii\data\DataProviderInterface $dataProvider', \Yii::$app->requestedParams['dataProvider']);
|
|
}
|
|
|
|
public function assertResponseStatus($status, $response): void
|
|
{
|
|
$this->assertInstanceOf('yii\console\Response', $response);
|
|
$this->assertSame($status, $response->exitStatus);
|
|
}
|
|
|
|
public function runRequest($route, $args = 0): Response
|
|
{
|
|
$request = new Request();
|
|
$request->setParams(func_get_args());
|
|
return Yii::$app->handleRequest($request);
|
|
}
|
|
|
|
public function testResponse(): void
|
|
{
|
|
$status = 123;
|
|
|
|
$response = $this->runRequest('fake/status');
|
|
$this->assertResponseStatus(0, $response);
|
|
|
|
$response = $this->runRequest('fake/status', (string)$status);
|
|
$this->assertResponseStatus($status, $response);
|
|
|
|
$response = $this->runRequest('fake/response');
|
|
$this->assertResponseStatus(0, $response);
|
|
|
|
$response = $this->runRequest('fake/response', (string)$status);
|
|
$this->assertResponseStatus($status, $response);
|
|
}
|
|
|
|
/**
|
|
* @see https://github.com/yiisoft/yii2/issues/12028
|
|
*/
|
|
public function testHelpOptionNotSet(): void
|
|
{
|
|
$controller = new FakeController('posts', Yii::$app);
|
|
$controller->runAction('index');
|
|
|
|
$this->assertTrue(FakeController::getWasActionIndexCalled());
|
|
$this->assertNull(FakeHelpController::getActionIndexLastCallParams());
|
|
}
|
|
|
|
/**
|
|
* @see https://github.com/yiisoft/yii2/issues/12028
|
|
*/
|
|
public function testHelpOption(): void
|
|
{
|
|
$controller = new FakeController('posts', Yii::$app);
|
|
$controller->help = true;
|
|
$controller->runAction('index');
|
|
|
|
$this->assertFalse(FakeController::getWasActionIndexCalled());
|
|
$this->assertEquals(FakeHelpController::getActionIndexLastCallParams(), ['posts/index']);
|
|
|
|
$helpController = new FakeHelpControllerWithoutOutput('help', Yii::$app);
|
|
$helpController->actionIndex('fake/aksi1');
|
|
$this->assertStringContainsString('--test-array, -ta', $helpController->outputString);
|
|
}
|
|
|
|
/**
|
|
* @see https://github.com/yiisoft/yii2/issues/13071
|
|
*/
|
|
public function testHelpOptionWithModule(): void
|
|
{
|
|
$controller = new FakeController('posts', new Module('news'));
|
|
$controller->help = true;
|
|
$controller->runAction('index');
|
|
|
|
$this->assertFalse(FakeController::getWasActionIndexCalled());
|
|
$this->assertEquals(FakeHelpController::getActionIndexLastCallParams(), ['news/posts/index']);
|
|
}
|
|
|
|
/**
|
|
* @see https://github.com/yiisoft/yii2/issues/19028
|
|
*/
|
|
public function testGetActionArgsHelp(): void
|
|
{
|
|
$controller = new FakeController('fake', Yii::$app);
|
|
$help = $controller->getActionArgsHelp($controller->createAction('aksi2'));
|
|
|
|
$this->assertArrayHasKey('values', $help);
|
|
$this->assertEquals('array', $help['values']['type']);
|
|
$this->assertArrayHasKey('value', $help);
|
|
// PHPDoc type
|
|
$this->assertEquals('string', $help['value']['type']);
|
|
}
|
|
|
|
public function testGetActionHelpSummaryOnNull(): void
|
|
{
|
|
$controller = new FakeController('fake', Yii::$app);
|
|
|
|
$controller->color = false;
|
|
$helpSummary = $controller->getActionHelpSummary(null);
|
|
$this->assertEquals('Action not found.', $helpSummary);
|
|
|
|
$controller->color = true;
|
|
$helpSummary = $controller->getActionHelpSummary(null);
|
|
$this->assertEquals($controller->ansiFormat('Action not found.', Console::FG_RED), $helpSummary);
|
|
}
|
|
}
|