From 5abe83e323f2d576d37a649ddf1daab24037a4b4 Mon Sep 17 00:00:00 2001 From: Brandon Kelly Date: Mon, 19 Aug 2024 01:30:57 -0400 Subject: [PATCH] Fix #20247: Support for variadic console controller action methods --- framework/CHANGELOG.md | 1 + framework/console/Controller.php | 28 +++++++++++++++------- tests/framework/console/ControllerTest.php | 6 +++++ tests/framework/console/FakeController.php | 5 ++++ 4 files changed, 32 insertions(+), 8 deletions(-) diff --git a/framework/CHANGELOG.md b/framework/CHANGELOG.md index c100209e75a..55f9605adfc 100644 --- a/framework/CHANGELOG.md +++ b/framework/CHANGELOG.md @@ -6,6 +6,7 @@ Yii Framework 2 Change Log - Bug #20232: Fix regression introduced in `GHSA-cjcc-p67m-7qxm` while attaching behavior defined by `__class` array key (erickskrauch) - Bug #20231: Fix regression introduced in #20167 in `yii\validators\FileValidator` (bizley) +- Enh #20247: Support for variadic console controller action methods (brandonkelly) 2.0.51 July 18, 2024 -------------------- diff --git a/framework/console/Controller.php b/framework/console/Controller.php index 908e50fcdb4..7029abd5462 100644 --- a/framework/console/Controller.php +++ b/framework/console/Controller.php @@ -198,6 +198,7 @@ public function bindActionParams($action, $params) $method = new \ReflectionMethod($action, 'run'); } + $paramKeys = array_keys($params); $args = []; $missing = []; $actionParams = []; @@ -212,16 +213,27 @@ public function bindActionParams($action, $params) } if ($key !== null) { - if (PHP_VERSION_ID >= 80000) { - $isArray = ($type = $param->getType()) instanceof \ReflectionNamedType && $type->getName() === 'array'; + if ($param->isVariadic()) { + for ($j = array_search($key, $paramKeys); $j < count($paramKeys); $j++) { + $jKey = $paramKeys[$j]; + if ($jKey !== $key && !is_int($jKey)) { + break; + } + $args[] = $actionParams[$key][] = $params[$jKey]; + unset($params[$jKey]); + } } else { - $isArray = $param->isArray(); - } - if ($isArray) { - $params[$key] = $params[$key] === '' ? [] : preg_split('/\s*,\s*/', $params[$key]); + if (PHP_VERSION_ID >= 80000) { + $isArray = ($type = $param->getType()) instanceof \ReflectionNamedType && $type->getName() === 'array'; + } else { + $isArray = $param->isArray(); + } + if ($isArray) { + $params[$key] = $params[$key] === '' ? [] : preg_split('/\s*,\s*/', $params[$key]); + } + $args[] = $actionParams[$key] = $params[$key]; + unset($params[$key]); } - $args[] = $actionParams[$key] = $params[$key]; - unset($params[$key]); } elseif ( PHP_VERSION_ID >= 70100 && ($type = $param->getType()) !== null diff --git a/tests/framework/console/ControllerTest.php b/tests/framework/console/ControllerTest.php index 2103715b51c..b3dd0350baa 100644 --- a/tests/framework/console/ControllerTest.php +++ b/tests/framework/console/ControllerTest.php @@ -91,6 +91,12 @@ public function testBindActionParams() $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('yii\console\Exception'); diff --git a/tests/framework/console/FakeController.php b/tests/framework/console/FakeController.php index 22158049d29..cc268c4b188 100644 --- a/tests/framework/console/FakeController.php +++ b/tests/framework/console/FakeController.php @@ -104,4 +104,9 @@ public function actionResponse($status = 0) $response->exitStatus = (int) $status; return $response; } + + public function actionVariadic($foo, $bar, ...$baz) + { + return [$foo, $bar, $baz]; + } }