diff --git a/docs/_data/menu.yml b/docs/_data/menu.yml index 570a836bc0dd386361c9b796b7f1da5cb85f12e7..8ec421f3d7addaa39cbc0d1f84b5998af49fecfd 100644 --- a/docs/_data/menu.yml +++ b/docs/_data/menu.yml @@ -6,5 +6,7 @@ Options: /options.html Operands: /operands.html Commands: /commands.html Help Text: /help.html +Localization: /localization.html +Validation: /validation.html Example: /example.html Development: /development.html diff --git a/docs/help.md b/docs/help.md index 8322585f61a7201458bf42ce2fb40f223fad18e3..e1ff564ad6d145193abe0a322c3a7ba63c5039e6 100644 --- a/docs/help.md +++ b/docs/help.md @@ -19,38 +19,13 @@ The method `HelpInterface::render(GetOpt, array)` receives the `GetOpt` object f ### Localization -By default, `GetOpt` displays standard help text in English. This can be customized in several ways. - -#### Switching to an existing language - -`GetOpt` comes bundled with help texts translated in -[several languages](https://github.com/getopt-php/getopt-php/tree/master/resources/localization). -These are located under `vendor/ulrichsg/getopt-php/resources/localization/<lang>.php`. - -You can switch to one of these languages by calling `GetOpt\GetOpt::setHelpLang($language)` - -```php -<?php -$getopt = new \GetOpt\GetOpt(); -$getopt->setHelpLang('de'); -``` - -Translations for additional languages are welcome; if you would like to contribute, please -[submit a pull request](https://github.com/getopt-php/getopt-php/compare). - -#### Switching to a custom language - -It is also possible to use a custom language file by specifying its path; the script must return an array in the -same format as the bundled language files. - -```php -<?php -$getopt = new \GetOpt\GetOpt(); -$getopt->setHelpLang(__DIR__ . '/path/to/cn.php'); -``` +By default, `GetOpt` displays standard help text in English. Have a look at [Localization](localization.md) for more +information about localization. #### Override the localization +> **Deprecated** note that this method is deprecated for messages. Please use localization files instead. + The `GetOpt\Help` class can be used to define the standard help text, with the `setTexts(array $texts)` method. The provided array overwrites the existing localization: @@ -108,6 +83,19 @@ $getopt->getHelp() ->setCommandsTemplate('path/to/my/commandsTemplate.php'); ``` +### Tables + +Options, operands and commands are shown in a table with two columns. The columns are divided with two spaces and the +description (second column) will automatically break after the last space that fits into the terminal. The number of +columns that fit into the terminal is determined in the following sequence: + +1. a constant `COLUMNS`, +2. an environment variable `COLUMNS`, +3. the result from `tput cols` command +4. value `90` + +This is limited by the setting `GetOpt\Help::MAX_WIDTH` (default: `120`). + ### The Parts of Help In the following sections, you will find a complete description of the three parts the Help is split into, and what @@ -124,21 +112,25 @@ command has to be given, where the options should be entered and the name and or `Usage: path/to/app make:config [options] <file>` - No commands, options and operands defined and strict operands: `Usage: path/to/app` + +#### Operands -#### Options +When one of the operands has a description the table of operands will be shown. The table can be suppressed by passing +the option `HIDE_OPERANDS` with a falsy value (e. g. `false`). -Options are shown in a table with the options (including argument) in the left column, and -the description of each option in the right column. +The table shows the operands name in the left column and the description on the right column what might look like this: -Long descriptions automatically break after the last space that fits into the -terminal's width. The number of columns is determined in the following sequence: +``` +Operands: + <source> The source that should be copied. + [<target>] The target where source should be copied to. If target is ommitted the file + will be copied to the current working directory. +``` -1. a constant `COLUMNS`, -2. an environment variable `COLUMNS`, -3. the result from `tput cols` command -4. value `90` +#### Options -This is limited by the setting `GetOpt\Help::MAX_WIDTH` (default: `120`). +Options are shown in a table with the options (including argument) in the left column, and +the description of each option in the right column. In the end it might look something like this: diff --git a/docs/localization.md b/docs/localization.md new file mode 100644 index 0000000000000000000000000000000000000000..4779e10ac1d2864c2ffcc2d20d7254cfcb4ab173 --- /dev/null +++ b/docs/localization.md @@ -0,0 +1,34 @@ +--- +layout: default +title: Localization +permalink: /localization.html +--- +# {{ page.title }} + +Texts shown to a user are localized using the `Translator` which is a simple key-value translation class using php files +for a translation map. These files can be found in [resources/localization](https://github.com/getopt-php/getopt-php/tree/master/resources/localization) +and contributions are highly appreciated. + +By default the Translator is using the English translations. You can change to a predefined translation by calling +`GetOpt::setLang($language)` where `$language` is a two-letter iso code of the language. + +```php +<?php +\GetOpt\GetOpt::setLang('de'); +``` + +## Custom language files + +You can also use custom language files by passing a path to your language file. The script must return an array in the +same format as the bundled language files. + +```php +<?php +\GetOpt\GetOpt::setLang(__DIR__ . '/path/to/cn.php'); +``` + +Translations missing in a language file are translated with the English translation table. + +## Errors messages + +The messages of exceptions thrown because of user input are also translated using this translator. diff --git a/docs/operands.md b/docs/operands.md index ddc0ec2949e8a92028326937d6f6d24f8d8ffb86..4b9704d22064816d10186cddb8dc4318d6dc1ebd 100644 --- a/docs/operands.md +++ b/docs/operands.md @@ -66,21 +66,15 @@ you are planning such things you should consider using `->getOperand('operand1') ### Validation -Again: it is the same functionality as for validating options. It follows a small example. See -[Options Validation](options.html#validation) for more details. +You can validate the argument of an operand using the `->setValidation($callable)`. To learn more about validation +please refer to the section [Validation](validation.md) of this handbook. -```php -<?php -$getopt = new \GetOpt\GetOpt(); -$getopt->addOperands([ - \GetOpt\Operand::create('file', \GetOpt\Operand::REQUIRED) - ->setValidation('is_readable'), - \GetOpt\Operand::create('destination', \GetOpt\Operand::MULTIPLE) - ->setValidation(function ($value) { - return file_exists($value) && is_dir($value) && is_writeable($value); - }), -]); -``` +### Description + +Since version 3.2 you can also set the description of operands with `->setDescription($description)`. When one of the +operands has a description the table of operands will be shown in the help. + +> **Note:** all operands will be shown even if they don't have a description to show the order of operands. ## Working With Operands diff --git a/docs/options.md b/docs/options.md index b4df566ec1f4baf4ad29bb3aa1b2a47bcaedf8de..179d03a1994913785abfdc2bf62d59bc728e4c81 100644 --- a/docs/options.md +++ b/docs/options.md @@ -275,54 +275,8 @@ var_dump($getopt->getOption('domain')); // ['example.com', 'example.org'] ### Validation -This library does not come with a bunch of validators that you can use and extend. Instead you provide a callable or -closure that has to return a truthy value if the value is valid (further called the validator). - -The validator gets the value as first and only parameter. For a lot of php standard functions this is enough (eg. -`is_numeric`). The value will always be a string or null. Here comes an example that shows how to check that it has -a valid json value: - -```php -<?php -$getopt = new \GetOpt\GetOpt([ - \GetOpt\Option::create(null, 'data', \GetOpt\GetOpt::REQUIRED_ARGUMENT) - ->setValidation(function ($value) { - return $value === 'null' || json_decode($value) !== null; - }) -]); -``` - -```console -$ php program.php --data null -$ php program.php --data [] -$ php program.php --data '{"a":"alpha"}' -$ php program.php --data invalid -``` - -#### Advanced Validation - -The validator is also executed if the option mode is `NO_ARGUMENT`. This way we can also check other circumstances -inside our application as well as the current status of options. - -A use case for this could be to define exclusive options (which is also the reason because it was asked in a feature -request). Let's say our program has the options `alpha` and `omega` but when you define `alpha` the `omega` option is -forbidden and vise versa: - -```php -<?php -$getopt = new \GetOpt\GetOpt(); - -$getopt->addOptions([ - \GetOpt\Option::create(null, 'alpha', \GetOpt\GetOpt::REQUIRED_ARGUMENT) - ->setValidation(function () use ($getopt) { - return !$getopt->getOption('omega'); - }), - \GetOpt\Option::create(null, 'omega', \GetOpt\GetOpt::REQUIRED_ARGUMENT) - ->setValidation(function () use ($getopt) { - return !$getopt->getOption('alpha'); - }), -]); -``` +You can validate the argument of an option using the `->setValidation($callable)`. To learn more about validation +please refer to the section [Validation](validation.md) of this handbook. ## Allow Custom Options diff --git a/docs/validation.md b/docs/validation.md new file mode 100644 index 0000000000000000000000000000000000000000..576c0db1bc218f1d357cfa68f538b48454b41a0c --- /dev/null +++ b/docs/validation.md @@ -0,0 +1,98 @@ +--- +layout: default +title: Validation +permalink: /validation.html +--- +# {{ page.title }} + +This library does not come with a bunch of validators that you can use and extend. Instead you provide a callable or +closure that has to return a truthy value if the value is valid (further called the validator). + +The validator gets the value as first and only parameter. For a lot of php standard functions this is enough (eg. +`is_numeric`). The value will always be a string or null. Here comes an example that shows how to check that it has +a valid json value: + +```php +<?php +$getopt = new \GetOpt\GetOpt([ + \GetOpt\Option::create(null, 'data', \GetOpt\GetOpt::REQUIRED_ARGUMENT) + ->setValidation(function ($value) { + return $value === 'null' || json_decode($value) !== null; + }) +]); +``` + +```console +$ php program.php --data null +$ php program.php --data [] +$ php program.php --data '{"a":"alpha"}' +$ php program.php --data invalid +``` + +## Validation Message + +Since version 3.2 GetOpt supports custom validation messages. `Option::setValidation()` and `Operand::setValidation` now +allow to pass a second parameter the validation message. This message gets passed to `sprintf($message, $desc, $value)` +where `$desc` is the description of the validated object (e. g. `Option 'data'`) and `$value` the original value that +was not valid. + +The validation message can also be a callable that is then called with `$object` and `$value`. To get the description +you can then use `$object->describe()`. + +```php +<?php +\GetOpt\Option::create('n', 'count', \GetOpt\GetOpt::REQUIRED_ARGUMENT) + ->setValidation(function ($value) { + return is_numeric($value) && (int)$value == round((double)$value); + }, 'Count has to be an integer'); +\GetOpt\Operand::create('pretend') + ->setValidation('is_resource', function(\GetOpt\Operand $operand, $value) { + return $value . ' is invalid for ' . $operand->getName(); + }); +``` + +## Reusable Validator + +You can also create functions or classes that can be used as validator and reuse them. + +```php +<?php +class InArrayValidator { + protected $array; + public function __construct(array $array) { + $this->array = $array; + } + public function __invoke($value) { + return in_array($value, $this->array); + } +} + +\GetOpt\Operand::create('type')->setValidation( + new InArrayValidator(['file', 'f', 'directory', 'd', 'link', 'l']) +); +``` + +## Advanced Validation + +The validator is also executed if the option mode is `NO_ARGUMENT`. This way we can also check other circumstances +inside our application as well as the current status of options. + +A use case for this could be to define exclusive options (which is also the reason because it was asked in a feature +request). Let's say our program has the options `alpha` and `omega` but when you define `alpha` the `omega` option is +forbidden and vise versa: + +```php +<?php +$getopt = new \GetOpt\GetOpt(); + +$getopt->addOptions([ + \GetOpt\Option::create(null, 'alpha', \GetOpt\GetOpt::REQUIRED_ARGUMENT) + ->setValidation(function () use ($getopt) { + return !$getopt->getOption('omega'); + }), + \GetOpt\Option::create(null, 'omega', \GetOpt\GetOpt::REQUIRED_ARGUMENT) + ->setValidation(function () use ($getopt) { + return !$getopt->getOption('alpha'); + }), +]); +``` diff --git a/resources/localization/de.php b/resources/localization/de.php index e10f5a67f11bd245850ed354909a6806db5fbf4c..e36965d19b8af4ab54e28b1172b2a1e86f5e8311 100644 --- a/resources/localization/de.php +++ b/resources/localization/de.php @@ -12,7 +12,7 @@ return [ 'no-more-operands' => 'Unerwarteter Operand %s', 'operand-missing' => 'Operand %s muss angegeben werden', 'option-argument-missing' => 'Option \'%s\' erwartet einen Wert', - 'option-value-invalid' => 'Der Wert für Option \'%s\' ist ungültig', + 'value-invalid' => 'Der Wert für %s ist ungültig', 'option' => 'Option', 'operand' => 'Operand', 'argument' => 'Argument', diff --git a/resources/localization/en.php b/resources/localization/en.php index 89bc65e22ee37631fabd3cec3d4b6d9417de4870..e23a5fd1ba26d7154491d044547271dad718d8fe 100644 --- a/resources/localization/en.php +++ b/resources/localization/en.php @@ -12,5 +12,5 @@ return [ 'no-more-operands' => 'No more operands expected - got %s', 'operand-missing' => 'Operand %s is required', 'option-argument-missing' => 'Option \'%s\' must have a value', - 'option-value-invalid' => 'Option \'%s\' has an invalid value', + 'value-invalid' => '%s has an invalid value', ]; diff --git a/resources/localization/fr.php b/resources/localization/fr.php index e696580774e58b6a1ae7f0ca9d60a18f9dad6946..2df830b06bd9002a31eec94222f2308da546a66d 100644 --- a/resources/localization/fr.php +++ b/resources/localization/fr.php @@ -12,7 +12,7 @@ return [ 'no-more-operands' => 'Argument inattendu - %s', 'operand-missing' => 'L\'argument %s est obligatoire', 'option-argument-missing' => 'L\'option \'%s\' doit avoir une valeur', - 'option-value-invalid' => 'La valeur de l\'option \'%s\' est invalide', + 'value-invalid' => 'La valeur de %s est invalide', 'option' => 'option', // optimization: otherwise we would have to load the fallback 'argument' => 'argument', 'operand' => 'argument', diff --git a/src/Argument.php b/src/Argument.php index 84af1aa683656071f5939bc25003240490b32b3b..607b97f0204a322d6f8e61dba34ac791e30f9d5b 100644 --- a/src/Argument.php +++ b/src/Argument.php @@ -104,7 +104,7 @@ class Argument implements Describable } return ucfirst(sprintf( - $this->validationMessage ?: '%s has an invalid value', + $this->validationMessage ?: GetOpt::translate('value-invalid'), $this->describe(), $value ));