我想知道如何从浏览器查询或控制器运行Symfony 2命令。
因为我没有任何可能在托管运行它,每个cron作业都由管理员设置。
我甚至没有启用exec()
功能所以当我想测试它时,我必须将所有内容从命令复制到一些测试控制器,这不是最好的解决方案。
有关较新版本的Symfony,请参阅official documentation
你不需要从控制器执行命令的服务,我认为最好通过run
方法调用命令而不是通过控制台字符串输入,但official docs建议你通过它的别名调用命令。另外,请参阅this answer。在Symfony 2.1-2.6上测试。
您的命令类必须扩展ContainerAwareCommand
// Your command
use Symfony\Bundle\FrameworkBundle\Command\ContainerAwareCommand;
class MyCommand extends ContainerAwareCommand {
// …
}
// Your controller
use Symfony\Component\Console\Input\ArrayInput;
use Symfony\Component\Console\Output\NullOutput;
class SomeController extends Controller {
// …
public function myAction()
{
$command = new MyCommand();
$command->setContainer($this->container);
$input = new ArrayInput(array('some-param' => 10, '--some-option' => true));
$output = new NullOutput();
$resultCode = $command->run($input, $output);
}
}
在大多数情况下,你不需要BufferedOutput
(来自Jbm的答案),它足以检查$resultCode is 0
,否则出现错误。
将您的命令注册为服务,不要忘记致电setContainer
MyCommandService:
class: MyBundle\Command\MyCommand
calls:
- [setContainer, ["@service_container"] ]
在您的控制器中,您只需要获取此服务,并使用权限参数调用execute方法
使用setArgument
方法设置输入:
$input = new Symfony\Component\Console\Input\ArgvInput([]);
$input->setArgument('arg1', 'value');
$output = new Symfony\Component\Console\Output\ConsoleOutput();
调用命令的run
方法:
$command = $this->get('MyCommandService');
$command->run($input, $output);
在我的环境(Symony 2.1)中,我必须对@Reuven解决方案进行一些修改才能使其正常工作。他们来了:
服务定义 - 无变化。
在控制器中:
use Symfony\Component\Console\Input\ArgvInput;
use Symfony\Component\Console\Output\ConsoleOutput;
...
public function myAction() {
$command = $this->get('MyCommandService');
$input = new ArgvInput(array('arg1'=> 'value'));
$output = new ConsoleOutput();
$command->run($input, $output);
}
这是一个替代方案,它允许您以与在控制台上相同的方式执行命令作为字符串(不需要使用此方法定义服务)。
您可以查看this bundle's controller,看看它是如何完成所有细节的。在这里,我将总结它省略某些细节(例如处理环境,所以这里所有命令都将在它们被调用的相同环境中运行)。
如果您只想从浏览器运行命令,可以按原样使用该包,但如果您想从任意控制器运行命令,请执行以下操作:
在你的控制器中定义一个这样的函数:
use Symfony\Bundle\FrameworkBundle\Console\Application;
use Symfony\Component\Console\Input\StringInput;
private function execute($command)
{
$app = new Application($this->get('kernel'));
$app->setAutoExit(false);
$input = new StringInput($command);
$output = new BufferedOutput();
$error = $app->run($input, $output);
if($error != 0)
$msg = "Error: $error";
else
$msg = $output->getBuffer();
return $msg;
}
然后你可以从这样的动作调用它:
public function dumpassetsAction()
{
$output = $this->execute('assetic:dump');
return new Response($output);
}
此外,您需要定义一个类作为输出缓冲区,因为框架没有提供:
use Symfony\Component\Console\Output\Output;
class BufferedOutput extends Output
{
public function doWrite($message, $newline)
{
$this->buffer .= $message. ($newline? PHP_EOL: '');
}
public function getBuffer()
{
return $this->buffer;
}
}
您只需创建一个命令实例并运行它:
/**
* @Route("/run-command")
*/
public function someAction()
{
// Running the command
$command = new YourCommand();
$command->setContainer($this->container);
$input = new ArrayInput(['--your_argument' => true]);
$output = new ConsoleOutput();
$command->run($input, $output);
return new Response();
}
和@malloc一样但是
use Symfony\Component\Console\Input\ArgvInput;
use Symfony\Component\Console\Output\ConsoleOutput;
...
public function myAction() {
$command = $this->get('MyCommandService');
// $input[0] : command name
// $input[1] : argument1
$input = new ArgvInput(array('my:command', 'arg1'));
$output = new ConsoleOutput();
$command->run($input, $output);
}
如果必须传递参数(和/或选项),那么在v2.0.12中(对于更高版本可能都是如此),您需要在实例化输入对象之前首先指定InputDefinition。
use // you will need the following
Symfony\Component\Console\Input\InputOption,
Symfony\Component\Console\Input\InputArgument,
Symfony\Component\Console\Input\InputDefinition,
Symfony\Component\Console\Input\ArgvInput,
Symfony\Component\Console\Output\NullOutput;
// tell symfony what to expect in the input
$inputDefinition = new InputDefinition(array(
new InputArgument('myArg1', InputArgument::REQUIRED),
new InputArgument('myArg2', InputArgument::REQUIRED),
new InputOption('debug', '0', InputOption::VALUE_OPTIONAL),
));
// then pass the values for arguments to constructor, however make sure
// first param is dummy value (there is an array_shift() in ArgvInput's constructor)
$input = new ArgvInput(
array(
'dummySoInputValidates' => 'dummy',
'myArg2' => 'myValue1',
'myArg2' => 'myValue2'),
$inputDefinition);
$output = new NullOutput();
作为旁注,如果您在命令中使用getContainer(),那么以下函数可能对您的command.php很方便:
/**
* Inject a dependency injection container, this is used when using the
* command as a service
*
*/
function setContainer(\Symfony\Component\DependencyInjection\ContainerInterface $container = null)
{
$this->container = $container;
}
/**
* Since we are using command as a service, getContainer() is not available
* hence we need to pass the container (via services.yml) and use this function to switch
* between conatiners..
*
*/
public function getcontainer()
{
if (is_object($this->container))
return $this->container;
return parent::getcontainer();
}
您可以使用此捆绑包从控制器(http请求)运行Symfony2命令,并在URL中传递选项/参数。
如果你运行一个需要像env
这样的assetic:dump
选项的命令
$stdout->writeln(sprintf('Dumping all <comment>%s</comment> assets.', $input->getOption('env')));
你必须创建一个Symfony\Component\Console\Application
并设置如下定义:
use Symfony\Component\Console\Application;
use Symfony\Component\Console\Input\ArgvInput;
use Symfony\Component\Console\Input\InputDefinition;
use Symfony\Component\Console\Input\InputOption;
use Symfony\Component\Console\Output\NullOuput;
// Create and run the command of assetic
$app = new Application();
$app->setDefinition(new InputDefinition([
new InputOption('env', '', InputOption::VALUE_OPTIONAL, '', 'prod')
]));
$app->add(new DumpCommand());
/** @var DumpCommand $command */
$command = $app->find('assetic:dump');
$command->setContainer($this->container);
$input = new ArgvInput([
'command' => 'assetic:dump',
'write_to' => $this->assetsDir
]);
$output = new NullOutput();
$command->run($input, $output);
您不能将选项env
设置为该命令,因为它不在其定义中。