Slide 1

Slide 1 text

Daniel Gomes @danielcsgomes October 17, 2013 icon by Austin Andrews © http://thenounproject.com/Templarian How to Build Console Applications

Slide 2

Slide 2 text

• Software Engineer @ GuestCentric Systems • ZCE PHP 5.3, OCP MySQL 5 Developer, CSM • @danielcsgomes about me

Slide 3

Slide 3 text

Lots of scripts Too many di!erent tools Tedious and boring

Slide 4

Slide 4 text

Centralize your tools Have interactive tools Easy to use by everyone Usage manual what about

Slide 5

Slide 5 text

Meet symfony Console Component

Slide 6

Slide 6 text

• Application • Command • Input • Output • Helpers • Events • Formatters console component toolset

Slide 7

Slide 7 text

APPLICATION COMMAND A COMMAND B COMMAND C plug-in & run

Slide 8

Slide 8 text

{ "require": { "symfony/console": "2.3.*@dev" } } install via composer zero dependencies

Slide 9

Slide 9 text

run(); Bootstrapping the application

Slide 10

Slide 10 text

$ php app.php

Slide 11

Slide 11 text

basic command

Slide 12

Slide 12 text

// Command/CacheClearCommand.php namespace DCSG\Command; use Symfony\Component\Console\Command\Command; class CacheClearCommand extends Command { protected function configure() { $this ->setName('cache:clear') ->setDescription('Clears Application Cache') ->setHelp(<<cache:clear command removes all files inside the cache directory. EOF ); } // ... }

Slide 13

Slide 13 text

add(new CacheClearCommand()); $app->run(); adding to application

Slide 14

Slide 14 text

$ php app.php

Slide 15

Slide 15 text

$ php app.php help cache:clear

Slide 16

Slide 16 text

add arguments

Slide 17

Slide 17 text

• ordered • input type: • optional • required • array arguments are:

Slide 18

Slide 18 text

// Command/CacheClearCommand.php $this->addArgument( 'directory', // name InputArgument::OPTIONAL, // input type 'The cache directory', // description './cache' // default value ); adding an argument

Slide 19

Slide 19 text

add options

Slide 20

Slide 20 text

• unordered • input type: • optional • required • array • none (no input) options are:

Slide 21

Slide 21 text

// CacheClearCommand.php $this->addOption( 'force', // name 'f', // shortname InputOption::VALUE_NONE, // input type 'Forces to clear the cache.' // description ); adding an option

Slide 22

Slide 22 text

add logic

Slide 23

Slide 23 text

// Command/CacheClearCommand.php protected function execute( InputInterface $input, OutputInterface $output) { // Logic here } execute the command

Slide 24

Slide 24 text

// Command/CacheClearCommand.php protected function execute(InputInterface $input, OutputInterface $output) { // Reads the Input $directory = $input->getArgument('directory'); // Removes all files inside the directory $counter = 0; $files = glob($cacheDir . '/*'); foreach ($files as $file) { if (is_file($file)) { unlink($file); $counter++; } } // Writes the Output $output->writeln(sprintf('Cache cleared. Removed %d files.', $counter)); }

Slide 25

Slide 25 text

always validate the input

Slide 26

Slide 26 text

// Command/CacheClearCommand.php // Reads the Input $directory = $input->getArgument('directory'); // Validate the argument input if (false === $cacheDir = realpath($directory)) { throw new \InvalidArgumentException('Directory does't exist.'); } // InputOption::VALUE_NONE does not need to be validate // since it does not accept parameters // in this case will return true or false $force = $input->getOption('force');

Slide 27

Slide 27 text

$ php app.php cache:clear /foo/bar

Slide 28

Slide 28 text

interact with the user

Slide 29

Slide 29 text

• Dialog helper • Formatter helper • Progress helper • Table helper

Slide 30

Slide 30 text

dialog HELPER

Slide 31

Slide 31 text

// Symfony/Component/Console/Helper/DialogHelper.php public function select(...){...} public function ask(...){...} public function askConfirmation(...){...} public function askHiddenResponse(...){...} public function askAndValidate(...){...} public function askHiddenResponseAndValidate (...){...}

Slide 32

Slide 32 text

// Command/CacheClearCommand.php // ... // Ask the user if it's the right directory $dialog = $this->getHelperSet()->get('dialog'); $answer = $dialog->askConfirmation( $output, sprintf('Remove all files in: %s ? (no)', $cacheDir), false // default value ); // ... Remove all files in: ./cache ? (no) yes OUTPUT

Slide 33

Slide 33 text

OUTPUT // examples/Helpers/select.php $colors = array('Red', 'Blue', 'Green', 'Yellow'); // Ask the user the favorite color $dialog = $this->getHelperSet()->get('dialog'); $colorIndex = $dialog->select( $output, 'What color do you like?', $colors ); // Writes the Output $output->writeln( sprintf('The color you like is %s.', $colors[$colorIndex]) );

Slide 34

Slide 34 text

formatter HELPER

Slide 35

Slide 35 text

OUTPUT $formatter = $this->getHelperSet()->get('formatter'); $formattedLine = $formatter->formatSection( 'SomeSection', 'Here is some message related to that section' ); $output->writeln($formattedLine); $msg = array('Something went wrong'); $formattedBlock = $formatter->formatBlock($msg, 'error'); $output->writeln($formattedBlock); $msg = array('Custom Colors'); $formattedBlock = $formatter->formatBlock($msg, 'bg=blue;fg=white'); $output->writeln($formattedBlock);

Slide 36

Slide 36 text

progress helper

Slide 37

Slide 37 text

05/50 [== ] 10% 25/50 [============= ] 50% 50/50 [============================] 100% $progress = $this->getHelperSet()->get('progress'); $progress->start($output, 50); $i = 0; while ($i++ < 50) { $progress->advance(); } $progress->finish(); OUTPUT

Slide 38

Slide 38 text

table helper

Slide 39

Slide 39 text

$table = $this->getHelperSet()->get('table'); $table ->setHeaders(array('Color', 'HEX')) ->setRows( array( array('Red', '#ff0000'), array('Blue', '#0000ff'), array('Green', '#008000'), array('Yellow', '#ffff00') ) ); $table->render($output); OUTPUT

Slide 40

Slide 40 text

command events

Slide 41

Slide 41 text

• command - before run • exception - on exceptions • terminate - before return exit code the events

Slide 42

Slide 42 text

testing Commands

Slide 43

Slide 43 text

public function testCacheClear() { $application = new Application(); $application->add(new CacheClearCommand()); // Mock the DialogHelper to skip interact // ... $command = $application->find('cache:clear'); // Add Mock to command $commandTester = new CommandTester($command); $commandTester->execute(array( 'command' => 'cache:clear', 'directory' => './cache' )); $this->assertFileNotExists($filename); }

Slide 44

Slide 44 text

// Symfony/Component/Console/Tester/CommandTester.php class CommandTester { private $command; private $input; private $output; public function __construct(Command $command){..} public function execute(array $input, array $options=array()){..} public function getDisplay($normalize = false){..} public function getInput(){..} public function getOutput(){..} } command tester

Slide 45

Slide 45 text

$ ./vendor/bin/phpunit

Slide 46

Slide 46 text

dependency injection in your application

Slide 47

Slide 47 text

Application Container CacheClearCommand FileSystem Finder Services setContainer(...) getContainer() addCommand() ... Other Commands

Slide 48

Slide 48 text

namespace Symfony\Component\DependencyInjection; interface ContainerAwareInterface { public function setContainer( ContainerInterface $container = null ); } container aware interface

Slide 49

Slide 49 text

// DCSG/Application.php class Application extends BaseApplication implements ContainerAwareInterface { private $container; public function setContainer(..){..} public function getContainer(){..} } implement in the application

Slide 50

Slide 50 text

// DCSG/Command/CacheClearCommand.php // ... $container = $this->getApplication()->getContainer(); $fs = $container->get('filesystem'); $finder = $container->get('finder'); $files = $finder->in($cacheDir); $fs->remove($files); // ... usage in the command

Slide 51

Slide 51 text

http://goo.gl/h0Xcfe http://symfony.com/doc/current/components/process.html Long running processes resources http://symfony.com/doc/current/components/console/index.html http://symfony.com/doc/current/cookbook/console/index.html http://symfony.com/doc/current/cookbook/service_container/index.html http://symfony.com/doc/current/components/dependency_injection/index.html Symfony2 Docs Cron jobs http://www.cyberciti.biz/faq/how-do-i-add-jobs-to-cron-under-linux-or-unix-oses/ Source Code https://github.com/danielcsgomes/phplx-how-to-build-console-app https://github.com/symfony/Console

Slide 52

Slide 52 text

@danielcsgomes | me@danielcsgomes.com | http://danielcsgomes.com Photo by Jian Awe © http://www."ickr.com/photos/qqjawe/6511141237 Please give feedback: https://joind.in/9615 Questions?