J4.x

Eine CLI-Anwendung erstellen

From Joomla! Documentation

This page is a translated version of the page J4.x:Writing A CLI Application and the translation is 17% complete.
Joomla! 
4.x
>Tutorial
Other languages:
Deutsch • ‎English • ‎中文(台灣)‎

Joomla 4 bringt eine aktualisierte Methode zur Erstellung von CLI-Befehlen mit. Wir werden versuchen, die wichtigsten Änderungen hier zu behandeln und erklären, wie man eigene CLI-Befehle in Joomla 4 erstellen kann.

Kurze Übersicht der Änderungen in Joomla 4

Wenn man in Joomla 4 eine CLI-Anwendung erstellt hat, musste man eine vollständige Anwendungsinstanz erstellen, indem man JApplicationCli erweiterte. In Joomla 4 haben wir, um eine neue Anwendungsinstanz zu erstellen, eine eigene Joomla-Anwendung erstellt, welche die Symfony-Konsolen-Komponente verwendet, um den Prozess der Integration Ihrer Komponenten wesentlich einfacher zu machen als bisher. Der Joomla 3 Weg, eine Anwendung durch JApplication zu erstellen, wird in Joomla 4 funktionieren, ist jedoch veraltet und wird in Joomla 5 entfernt werden.

The new Console Package (a new package in version 2 of the Joomla Framework) contains the base code and it's extended in the CMS to add support for things like using JInput.

The advantage of the new way is that we have a single application that means using JApplication::isClient('cli') will work. The introduction of \\Joomla\\CMS\\Application\\CMSApplicationInterface means that all applications within Joomla should have a consistent set of methods available for extensions to use.

Creating a Command

The most rudimentary command just prints a title and some text into the terminal.

	/**
	 * The default command name
	 *
	 * @var    string
	 * @since  4.0.0
	 */
	protected static $defaultName = 'commandgroup:commandname';

	/**
	 * Internal function to execute the command.
	 *
	 * @param   InputInterface   $input   The input to inject into the command.
	 * @param   OutputInterface  $output  The output to inject into the command.
	 *
	 * @return  integer  The command exit code
	 *
	 * @since   4.0.0
	 */
	protected function doExecute(InputInterface $input, OutputInterface $output): int
	{
		$symfonyStyle = new SymfonyStyle($input, $output);

		$symfonyStyle->title('Hello World Command Title');

		// You might want to do some stuff here in Joomla

		$symfonyStyle->success('Hello World!');

		return 0;
	}

	/**
	 * Configure the command.
	 *
	 * @return  void
	 *
	 * @since   4.0.0
	 */
	protected function configure(): void
	{
		$this->setDescription('This command prints hello world to whoever calls it');
		$this->setHelp(
			<<<EOF
The <info>%command.name%</info> command prints hello world to whoever calls it
<info>php %command.full_name%</info>
EOF
		);
	}

Let's cover the 3 parts of this code:

  • configure function: Configures information about the command setting up information in the class which is shown when a user lists help information for the command* doExecute function: This method does the work of the command - you'd normally go and execute a Joomla Component here (i.e. Dispatch a components dispatcher)* defaultName property: This is the command name that is run when executed (remember this we'll come back to it in a second!)If you need more help there's an excellent Video produced as part of our GSOC 2018 program to help you out!

Adding the command to the Application by Plugin

Inside your plugin you're going to register to the event \Joomla\Application\ApplicationEvents::BEFORE_EXECUTE using the method getSubscribedEvents. The following Example required a command RunHelloCommand and registers it.

class PlgsystemMyConsole extends CMSPlugin implements SubscriberInterface
{

    public static function getSubscribedEvents(): array
    {
        return [
            \Joomla\Application\ApplicationEvents::BEFORE_EXECUTE => 'registerCommands',
        ];
    }

    public function registerCommands(): void
    {
        $serviceId = 'hello.world';

        Factory::getContainer()->share(
            $serviceId,
            function (\Psr\Container\ContainerInterface $container) {
                // do stuff to create command class and return it
                return new RunHelloCommand();
            },
            true
        );

        Factory::getContainer()->get(\Joomla\CMS\Console\Loader\WritableLoaderInterface::class)->add('hello:world', $serviceId);
    }

}

This method registerCommands adds your command into the Dependency Injection Container then registers that command against the name of the command for when it's called by the Joomla CLI Application (note the first parameter of WritableLoaderInterface::add() should be the same as the defaultName property inside your Command Class.

Calling the Command

To call the command you just created you need to run it through the Joomla CLI Command:

php cli/joomla.php my:command