Как я могу запустить команду запуска Symfony 2 с контроллера
Мне интересно, как я могу запустить команду Symfony 2 из запроса браузера или из контроллера.
Это потому, что у меня нет никакой возможности запустить его на хостинге, и все задания cron устанавливаются администратором.
Я даже не включил exec()
Функция, поэтому, когда я хочу проверить это, я должен скопировать весь контент из команды на некоторый контроллер тестирования, и это не лучшее решение.
9 ответов
Смотрите официальную документацию по этому вопросу для новых версий Symfony
Вам не нужны сервисы для выполнения команд с контроллера, и, я думаю, лучше вызывать команду через run
метод, а не через консольный ввод строки, однако официальные документы предлагают вам вызывать команду через ее псевдоним. Также смотрите этот ответ. Проверено на 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, $ouput);
В моей среде ( 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);
}
Вы можете просто создать экземпляр вашей команды и запустить его:
/**
* @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();
}
Вот альтернатива, которая позволяет вам выполнять команды в виде строк так же, как на консоли (нет необходимости определять сервисы с этим).
Вы можете проверить контроллер этого пакета, чтобы увидеть, как это делается со всеми деталями. Здесь я собираюсь подвести итог, опуская некоторые детали (например, обработку среды, поэтому здесь все команды будут выполняться в той же среде, в которой они были вызваны).
Если вы хотите просто запускать команды из браузера, вы можете использовать этот пакет как есть, но если вы хотите запускать команды с произвольного контроллера, вот как это сделать:
В вашем контроллере определите такую функцию:
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;
}
}
Так же, как @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();
}
Вы можете использовать этот пакет для запуска команд Symfony2 из контроллера (запрос http) и передачи параметров / параметров в 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
к команде, потому что это не в ее определении.