diff --git a/DependencyInjection/Configuration.php b/DependencyInjection/Configuration.php index 557d96d..88cbe8b 100644 --- a/DependencyInjection/Configuration.php +++ b/DependencyInjection/Configuration.php @@ -53,6 +53,11 @@ class Configuration implements ConfigurationInterface ->end() ->end() ->end() + ->arrayNode('messenger') + ->children() + ->scalarNode('message_handler')->isRequired()->defaultValue('simple_console_runner')->end() + ->end() + ->end() ->end(); return $treeBuilder; diff --git a/DependencyInjection/RetailCrmServiceExtension.php b/DependencyInjection/RetailCrmServiceExtension.php index d32f08e..0c2998b 100644 --- a/DependencyInjection/RetailCrmServiceExtension.php +++ b/DependencyInjection/RetailCrmServiceExtension.php @@ -4,6 +4,7 @@ namespace RetailCrm\ServiceBundle\DependencyInjection; use RetailCrm\ServiceBundle\ArgumentResolver\CallbackValueResolver; use RetailCrm\ServiceBundle\ArgumentResolver\ClientValueResolver; +use RetailCrm\ServiceBundle\Messenger\MessageHandler; use RetailCrm\ServiceBundle\Response\ErrorJsonResponseFactory; use RetailCrm\ServiceBundle\Security\CallbackClientAuthenticator; use RetailCrm\ServiceBundle\Security\FrontApiClientAuthenticator; @@ -49,6 +50,11 @@ class RetailCrmServiceExtension extends Extension $config['request_schema']['client']['serializer'] ); + $container->setParameter( + 'retail_crm_service.messenger.message_handler', + $config['messenger']['message_handler'] + ); + $container ->register(SymfonySerializerAdapter::class) ->setAutowired(true); @@ -90,5 +96,23 @@ class RetailCrmServiceExtension extends Extension $container ->register(FrontApiClientAuthenticator::class) ->setAutowired(true); + + $container + ->register(MessageHandler\SimpleConsoleRunner::class) + ->setAutowired(true); + $container->setAlias('simple_console_runner', MessageHandler\SimpleConsoleRunner::class); + + $container + ->register(MessageHandler\InNewProcessRunner::class) + ->setAutowired(true); + $container->setAlias('in_new_process_runner', MessageHandler\InNewProcessRunner::class); + + $container + ->register(MessageHandler::class) + ->addTag('messenger.message_handler') + ->setArguments([ + new Reference($container->getParameter('retail_crm_service.messenger.message_handler')) + ]) + ->setAutowired(true); } } diff --git a/Messenger/Message.php b/Messenger/Message.php new file mode 100644 index 0000000..c4fdb23 --- /dev/null +++ b/Messenger/Message.php @@ -0,0 +1,99 @@ +commandName; + } + + /** + * @param string $commandName + */ + public function setCommandName(string $commandName): void + { + $this->commandName = $commandName; + } + + /** + * @return array + */ + public function getOptions(): array + { + return $this->options; + } + + /** + * @param array $options + */ + public function setOptions(array $options): void + { + $this->options = $options; + } + + /** + * @return array + */ + public function getArguments(): array + { + return $this->arguments; + } + + /** + * @param array $arguments + */ + public function setArguments(array $arguments): void + { + $this->arguments = $arguments; + } + + /** + * @param string $key + * @param string $value + */ + public function addOption(string $key, string $value): void + { + $this->options[$key] = $value; + } + + /** + * @param string $key + * @param string $value + */ + public function addArgument(string $key, string $value): void + { + $this->arguments[$key] = $value; + } + + /** + * @return array + */ + public function getFormattedOptions(): array + { + $options = []; + foreach ($this->options as $option => $value) { + $options['--' . $option] = $value; + } + + return $options; + } +} diff --git a/Messenger/MessageHandler.php b/Messenger/MessageHandler.php new file mode 100644 index 0000000..f8cc6ce --- /dev/null +++ b/Messenger/MessageHandler.php @@ -0,0 +1,40 @@ +runner = $runner; + } + + /** + * @param Message $message + * + * @throws Exception + */ + public function __invoke(Message $message): void + { + $this->runner->run($message); + } +} diff --git a/Messenger/MessageHandler/InNewProcessRunner.php b/Messenger/MessageHandler/InNewProcessRunner.php new file mode 100644 index 0000000..e200508 --- /dev/null +++ b/Messenger/MessageHandler/InNewProcessRunner.php @@ -0,0 +1,96 @@ +logger = $logger; + $this->kernel = $kernel; + } + + /** + * {@inheritdoc} + */ + public function run(Message $message): void + { + $phpBinaryPath = (new PhpExecutableFinder)->find(); + $consoleCommand = [ + 'php' => $phpBinaryPath ?: 'php', + 'console' => sprintf('%s/bin/console', $this->kernel->getContainer()->getParameter('kernel.project_dir')), + 'command' => $message->getCommandName() + ]; + + $process = new Process( + array_merge( + array_values($consoleCommand), + array_values($message->getArguments()), + array_values($this->getOptions($message)), + ) + ); + + try { + $process + ->setTimeout(static::DEFAULT_TIMEOUT) + ->run(static function(string $type, string $buffer) { + echo $buffer; + }) + ; + } catch (ProcessTimedOutException $processTimedOutException) { + $this->logger->error( + sprintf( + 'Process "%s" killed after %d seconds of execution', + $processTimedOutException->getProcess()->getCommandLine(), + $processTimedOutException->getProcess()->getTimeout() + ) + ); + } + } + + /** + * @param Message $message + * + * @return array + */ + private function getOptions(Message $message): array + { + $options = []; + foreach ($message->getFormattedOptions() as $option => $value) { + $options[] = $option . '=' . $value; + } + + return $options; + } +} diff --git a/Messenger/MessageHandler/JobRunner.php b/Messenger/MessageHandler/JobRunner.php new file mode 100644 index 0000000..bb3afe8 --- /dev/null +++ b/Messenger/MessageHandler/JobRunner.php @@ -0,0 +1,18 @@ +logger = $logger; + $this->kernel = $kernel; + } + + /** + * {@inheritdoc} + */ + public function run(Message $message): void + { + $application = new Application($this->kernel); + $application->setAutoExit(false); + + $input = new ArrayInput( + array_merge( + ['command' => $message->getCommandName()], + $message->getFormattedOptions(), + $message->getArguments() + ) + ); + + $output = new BufferedOutput(); + $application->run($input, $output); + + echo $output->fetch(); + } +} diff --git a/composer.json b/composer.json index cdfb4a4..44539dd 100644 --- a/composer.json +++ b/composer.json @@ -16,7 +16,10 @@ "symfony/serializer": "^5.2", "symfony/http-kernel": "^4.0|^5.0", "symfony/validator": "^4.0|^5.0", - "symfony/security-guard": "^4.0|^5.0" + "symfony/security-guard": "^4.0|^5.0", + "symfony/console": "^5.2", + "symfony/messenger": "^5.2", + "symfony/process": "^5.2" }, "autoload": { "psr-4": {