如何通过交互调用工匠控制台命令

Posted

技术标签:

【中文标题】如何通过交互调用工匠控制台命令【英文标题】:How to call an artisan console command with an interaction 【发布时间】:2016-03-15 22:21:06 【问题描述】:

我目前正在一个 Laravel 5.1 项目中创建一个 php artisan 控制台命令,并且想从我的控制台命令中调用另一个控制台命令。我要调用的这个第三方命令不接受任何选项或参数,而是通过交互式问题接收其输入。

我知道我可以使用如下选项和参数调用命令:

$this->call('command:name', ['argument' => 'foo', '--option' => 'bar']);

我也知道我可以从命令行调用交互式命令而无需像这样的交互:

php artisan command:name --no-interaction


但是如何在我的命令中回答这些交互式问题呢?

我想做类似下面的事情(伪代码)。

$this->call('command:name', [
    'argument' => 'foo', 
    '--option' => 'bar'
], function($console) 
    $console->writeln('Yes'); //answer an interactive question 
    $console-writeln('No'); //answer an interactive question 
    $console->writeln(''); //skip answering an interactive question 
 );

上面当然不行,因为$this->call($command, $arguments)不接受第三个回调参数。

从控制台命令调用控制台命令时如何回答交互式问题?

【问题讨论】:

我对这个问题很感兴趣。我已经在 Laravel 沙盒中玩了一个小时左右,看看我是否可以让它工作。我得出了一个不幸的结论,即实现此功能将是大量的工作。目前,无法通过在另一个调用中进行的调用传递任何内容,而无需通过参数提供默认值。 @DavidBarker 我通过在 Symfony 类方法 QuestionHelper@doAsk 中添加 5 行代码使其工作(现在)。我目前正在集思广益,考虑如何在不触及核心的情况下增强这种方法,考虑使用runkit_method_redefine()。如果/当我成功时会更新这篇文章。欢迎提出建议。 仍然非常老套,但如果没有 Symfony 组件中可用的功能,我想不出或看到任何其他方法来实现您需要的功能。 【参考方案1】:

与mpyw/streamable-console: Call interactive artisan command using arbitrary stream:

$this->usingInputStream("yes\nno\n")->call('command:name');

【讨论】:

【参考方案2】:

我有另一个解决方案,它是调用 symfony 命令执行 'php artisan' 而不是使用 artisan 子命令。我认为这比修补 3rd 方代码要好。

这是一个管理这个的特征。

use Symfony\Component\Process\Process;
use Symfony\Component\Process\Exception\ProcessFailedException;
trait ArtisanCommandTrait
    public function executeArtisanCommand($command, $options)
        $stmt = 'php artisan '. $command . ' ' . $this->prepareOptions($options);

        $process = new Process($stmt);
        $process->run();
        // executes after the command finishes
        if (!$process->isSuccessful()) 
            throw new ProcessFailedException($process);
        
        return $process->getOutput();
    
    public function prepareOptions($options)
            $args = [];
            $opts = [];
            $flags = [];
            foreach ($options as $key => $value) 
                if(ctype_alpha(substr($key, 0, 1)))
                    $args[] = $value;
                else if(starts_with($key, '--'))
                    $opts[] = $key. (is_null($value) ? '' : '=' . $value) ;
                
                else if(starts_with($key, '-'))
                    $flags[] = $key;
                
            
            return   implode(' ', $args) . ' '
                    .implode(' ', $opts). ' '
                    .implode(' ', $flags);
    

现在,您应该能够传递任何工匠特殊选项,例如无交互。

public function handle()
    $options = [
            'argument' => $argument,
            '--option' => $options,         // options should be preceded by --
            '-n' => null                    // no-interaction option
        ];
    $command = 'your:command';
    $output = $this->executeArtisanCommand($command, $options);        
    echo $output;

您可以从此gist下载特征

【讨论】:

这样更好;我接受你的回答是最好的 @PepijnOlivier 除非你没有【参考方案3】:

我是这样做的。

注意:这修补了核心 Symfony 类 QuestionHelper@doAsk,虽然这段代码可以正常运行(我目前只是做概念证明),但这段代码可能不应该在任何生产环境中运行。 我还没有接受我自己的答案,想知道是否有更好的方法来做到这一点。

以下假设安装了 Laravel 5.1。

首先作曲家需要 Patchwork 包。我正在使用它来增强 Symfony 类方法的功能。

composer require antecedent/patchwork

编辑bootstrap/app.php并在应用程序创建后立即添加以下内容。(不自动加载补丁)

if($app->runningInConsole()) 
    require_once(__DIR__ . '/../vendor/antecedent/patchwork/Patchwork.php');
;

将以下两个 use 语句添加到控制台命令类的顶部

use Symfony\Component\Console\Output\OutputInterface;

use Symfony\Component\Console\Question\Question;

通过在控制台命令类上使用这些辅助方法来增强/修补 QuestionHelper@doAsk

public function __construct() 
    parent::__construct();
    $this->patchAskingQuestion();


/**
 * Patch QuestionHelper@doAsk
 * When a key 'qh-patch-answers' is found in the $_REQUEST superglobal,
 * We assume this is an array which holds the answers for our interactive questions.
 * shift each answer off the array, before answering the corresponding question.
 * When an answer has a NULL value, we will just provide the default answer (= skip question)
 */
private function patchAskingQuestion() 

    \Patchwork\replace('Symfony\Component\Console\Helper\QuestionHelper::doAsk', function(OutputInterface $output, Question $question) 

        $answers = &$_REQUEST['qh-patch-answers'];

        //No predefined answer found? Just call the original method
        if(empty($answers)) 
            return \Patchwork\callOriginal([$output, $question]);
        

        //using the next predefined answer, or the default if the predefined answer was NULL
        $answer = array_shift($answers);
        return ($answer === null) ? $question->getDefault() : $answer;
    );


private function setPredefinedAnswers($answers) 
    $_REQUEST['qh-patch-answers'] = $answers;


private function clearPredefinedAnswers() 
    unset($_REQUEST['qh-patch-answers']);

您现在可以回答这样的互动问题

public function fire() 
    //predefine the answers to the interactive questions
    $this->setPredefinedAnswers([
        'Yes', //first question will be answered with 'Yes'
        'No', //second question will be answered with 'No'
        null, //third question will be skipped (using the default answer)
        null, //fourth question will be skipped (using the default answer)
    ]);

    //call the interactive command
    $this->call('command:name');

    //clean up, so future calls to QuestionHelper@doAsk will definitely call the original method
    $this->clearPredefinedAnswers();

【讨论】:

以上是关于如何通过交互调用工匠控制台命令的主要内容,如果未能解决你的问题,请参考以下文章

在我的控制器中运行工匠命令(make:migration)

如何从 Laravel Artisan 触发单元测试

使用一个工匠命令制作模型、迁移和控制器,但控制器需要在不同的目录中

Laravel 4 - PHP 标签在工匠控制台和渲染页面中添加前缀

Laravel - 从路由调用工匠队列命令 - 邮递员超时

如何通过 Perl 以编程方式控制交互式 Unix 应用程序?