CliOptionsParserTest.php 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. <?php
  2. declare(strict_types=1);
  3. use PHPUnit\Framework\TestCase;
  4. require_once dirname(__DIR__, 2) . '/cli/CliOption.php';
  5. require_once dirname(__DIR__, 2) . '/cli/CliOptionsParser.php';
  6. final class CliOptionsOptionalTest extends CliOptionsParser {
  7. public string $string = '';
  8. public int $int = 0;
  9. public bool $bool = false;
  10. /** @var array<int,string> $arrayOfString */
  11. public array $arrayOfString = [];
  12. public string $defaultInput = '';
  13. public string $optionalValue = '';
  14. public bool $optionalValueWithDefault = false;
  15. public string $defaultInputAndOptionalValueWithDefault = '';
  16. public bool $flag = false;
  17. public function __construct() {
  18. $this->addOption('string', (new CliOption('string', 's'))->deprecatedAs('deprecated-string'));
  19. $this->addOption('int', (new CliOption('int', 'i'))->typeOfInt());
  20. $this->addOption('bool', (new CliOption('bool', 'b'))->typeOfBool());
  21. $this->addOption('arrayOfString', (new CliOption('array-of-string', 'a'))->typeOfArrayOfString());
  22. $this->addOption('defaultInput', (new CliOption('default-input', 'i')), 'default');
  23. $this->addOption('optionalValue', (new CliOption('optional-value', 'o'))->withValueOptional());
  24. $this->addOption('optionalValueWithDefault', (new CliOption('optional-value-with-default', 'd'))->withValueOptional('true')->typeOfBool());
  25. $this->addOption('defaultInputAndOptionalValueWithDefault',
  26. (new CliOption('default-input-and-optional-value-with-default', 'e'))->withValueOptional('optional'),
  27. 'default'
  28. );
  29. $this->addOption('flag', (new CliOption('flag', 'f'))->withValueNone());
  30. parent::__construct();
  31. }
  32. }
  33. final class CliOptionsOptionalAndRequiredTest extends CliOptionsParser {
  34. public string $required = '';
  35. public string $string = '';
  36. public int $int = 0;
  37. public bool $bool = false;
  38. public bool $flag = false;
  39. public function __construct() {
  40. $this->addRequiredOption('required', new CliOption('required'));
  41. $this->addOption('string', new CliOption('string', 's'));
  42. $this->addOption('int', (new CliOption('int', 'i'))->typeOfInt());
  43. $this->addOption('bool', (new CliOption('bool', 'b'))->typeOfBool());
  44. $this->addOption('flag', (new CliOption('flag', 'f'))->withValueNone());
  45. parent::__construct();
  46. }
  47. }
  48. class CliOptionsParserTest extends TestCase {
  49. public static function testInvalidOptionSetWithValueReturnsError(): void {
  50. $result = self::runOptionalOptions('--invalid=invalid');
  51. self::assertSame(['invalid' => 'unknown option: invalid'], $result->errors);
  52. }
  53. public static function testInvalidOptionSetWithoutValueReturnsError(): void {
  54. $result = self::runOptionalOptions('--invalid');
  55. self::assertSame(['invalid' => 'unknown option: invalid'], $result->errors);
  56. }
  57. public static function testValidOptionSetWithValidValueAndInvalidOptionSetWithValueReturnsValueForValidOptionAndErrorForInvalidOption(): void {
  58. $result = self::runOptionalOptions('--string=string --invalid=invalid');
  59. self::assertSame('string', $result->string);
  60. self::assertSame(['invalid' => 'unknown option: invalid'], $result->errors);
  61. }
  62. public static function testOptionWithValueTypeOfStringSetOnceWithValidValueReturnsValueAsString(): void {
  63. $result = self::runOptionalOptions('--string=string');
  64. self::assertSame('string', $result->string);
  65. }
  66. public static function testOptionWithRequiredValueTypeOfIntSetOnceWithValidValueReturnsValueAsInt(): void {
  67. $result = self::runOptionalOptions('--int=111');
  68. self::assertSame(111, $result->int);
  69. }
  70. public static function testOptionWithRequiredValueTypeOfBoolSetOnceWithValidValueReturnsValueAsBool(): void {
  71. $result = self::runOptionalOptions('--bool=on');
  72. self::assertTrue($result->bool);
  73. }
  74. public static function testOptionWithValueTypeOfArrayOfStringSetOnceWithValidValueReturnsValueAsArrayOfString(): void {
  75. $result = self::runOptionalOptions('--array-of-string=string');
  76. self::assertSame(['string'], $result->arrayOfString);
  77. }
  78. public static function testOptionWithValueTypeOfStringSetMultipleTimesWithValidValueReturnsLastValueSetAsString(): void {
  79. $result = self::runOptionalOptions('--string=first --string=second');
  80. self::assertSame('second', $result->string);
  81. }
  82. public static function testOptionWithValueTypeOfIntSetMultipleTimesWithValidValueReturnsLastValueSetAsInt(): void {
  83. $result = self::runOptionalOptions('--int=111 --int=222');
  84. self::assertSame(222, $result->int);
  85. }
  86. public static function testOptionWithValueTypeOfBoolSetMultipleTimesWithValidValueReturnsLastValueSetAsBool(): void {
  87. $result = self::runOptionalOptions('--bool=on --bool=off');
  88. self::assertFalse($result->bool);
  89. }
  90. public static function testOptionWithValueTypeOfArrayOfStringSetMultipleTimesWithValidValueReturnsAllSetValuesAsArrayOfString(): void {
  91. $result = self::runOptionalOptions('--array-of-string=first --array-of-string=second');
  92. self::assertSame(['first', 'second'], $result->arrayOfString);
  93. }
  94. public static function testOptionWithValueTypeOfIntSetWithInvalidValueReturnsAnError(): void {
  95. $result = self::runOptionalOptions('--int=one');
  96. self::assertSame(['int' => 'invalid input: int must be an integer'], $result->errors);
  97. }
  98. public static function testOptionWithValueTypeOfBoolSetWithInvalidValuesReturnsAnError(): void {
  99. $result = self::runOptionalOptions('--bool=bad');
  100. self::assertSame(['bool' => 'invalid input: bool must be a boolean'], $result->errors);
  101. }
  102. public static function testOptionWithValueTypeOfIntSetMultipleTimesWithValidAndInvalidValuesReturnsLastValidValueSetAsIntAndError(): void {
  103. $result = self::runOptionalOptions('--int=111 --int=one --int=222 --int=two');
  104. self::assertSame(222, $result->int);
  105. self::assertSame(['int' => 'invalid input: int must be an integer'], $result->errors);
  106. }
  107. public static function testOptionWithValueTypeOfBoolSetMultipleTimesWithWithValidAndInvalidValuesReturnsLastValidValueSetAsBoolAndError(): void {
  108. $result = self::runOptionalOptions('--bool=on --bool=good --bool=off --bool=bad');
  109. self::assertFalse($result->bool);
  110. self::assertSame(['bool' => 'invalid input: bool must be a boolean'], $result->errors);
  111. }
  112. public static function testNotSetOptionWithDefaultInputReturnsDefaultInput(): void {
  113. $result = self::runOptionalOptions('');
  114. self::assertSame('default', $result->defaultInput);
  115. }
  116. public static function testOptionWithDefaultInputSetWithValidValueReturnsCorrectlyTypedValue(): void {
  117. $result = self::runOptionalOptions('--default-input=input');
  118. self::assertSame('input', $result->defaultInput);
  119. }
  120. public static function testOptionWithOptionalValueSetWithoutValueReturnsEmptyString(): void {
  121. $result = self::runOptionalOptions('--optional-value');
  122. self::assertSame('', $result->optionalValue);
  123. }
  124. public static function testOptionWithOptionalValueDefaultSetWithoutValueReturnsOptionalValueDefault(): void {
  125. $result = self::runOptionalOptions('--optional-value-with-default');
  126. self::assertTrue($result->optionalValueWithDefault);
  127. }
  128. public static function testNotSetOptionWithOptionalValueDefaultAndDefaultInputReturnsDefaultInput(): void {
  129. $result = self::runOptionalOptions('');
  130. self::assertSame('default', $result->defaultInputAndOptionalValueWithDefault);
  131. }
  132. public static function testOptionWithOptionalValueDefaultAndDefaultInputSetWithoutValueReturnsOptionalValueDefault(): void {
  133. $result = self::runOptionalOptions('--default-input-and-optional-value-with-default');
  134. self::assertSame('optional', $result->defaultInputAndOptionalValueWithDefault);
  135. }
  136. public static function testOptionWithFlag(): void {
  137. $result = self::runOptionalOptions('--flag');
  138. self::assertTrue($result->flag);
  139. }
  140. public static function testOptionWithNoFlag(): void {
  141. $result = self::runOptionalOptions('');
  142. self::assertFalse($result->flag);
  143. }
  144. public static function testRequiredOptionNotSetReturnsError(): void {
  145. $result = self::runOptionalAndRequiredOptions('');
  146. self::assertSame(['required' => 'invalid input: required cannot be empty'], $result->errors);
  147. }
  148. public static function testOptionSetWithDeprecatedAliasGeneratesDeprecationWarningAndReturnsValue(): void {
  149. $result = self::runCommandReadingStandardError('--deprecated-string=string');
  150. self::assertSame('FreshRSS deprecation warning: the CLI option(s): deprecated-string are deprecated ' .
  151. 'and will be removed in a future release. Use: string instead',
  152. $result
  153. );
  154. $result = self::runOptionalOptions('--deprecated-string=string');
  155. self::assertSame('string', $result->string);
  156. }
  157. public static function testAlwaysReturnUsageMessageWithUsageInfoForAllOptions(): void {
  158. $result = self::runOptionalAndRequiredOptions('');
  159. self::assertSame('Usage: cli-parser-test.php --required=<required> [-s --string=<string>] [-i --int=<int>] [-b --bool=<bool>] [-f --flag]',
  160. $result->usage,
  161. );
  162. }
  163. private static function runOptionalOptions(string $cliOptions = ''): CliOptionsOptionalTest {
  164. $command = __DIR__ . '/cli-parser-test.php';
  165. $className = CliOptionsOptionalTest::class;
  166. $result = shell_exec("CLI_PARSER_TEST_OPTIONS_CLASS='$className' $command $cliOptions 2>/dev/null");
  167. $result = is_string($result) ? unserialize($result) : new CliOptionsOptionalTest();
  168. /** @var CliOptionsOptionalTest $result */
  169. return $result;
  170. }
  171. private static function runOptionalAndRequiredOptions(string $cliOptions = ''): CliOptionsOptionalAndRequiredTest {
  172. $command = __DIR__ . '/cli-parser-test.php';
  173. $className = CliOptionsOptionalAndRequiredTest::class;
  174. $result = shell_exec("CLI_PARSER_TEST_OPTIONS_CLASS='$className' $command $cliOptions 2>/dev/null");
  175. $result = is_string($result) ? unserialize($result) : new CliOptionsOptionalAndRequiredTest();
  176. /** @var CliOptionsOptionalAndRequiredTest $result */
  177. return $result;
  178. }
  179. private static function runCommandReadingStandardError(string $cliOptions = ''): string {
  180. $command = __DIR__ . '/cli-parser-test.php';
  181. $className = CliOptionsOptionalTest::class;
  182. $result = shell_exec("CLI_PARSER_TEST_OPTIONS_CLASS='$className' $command $cliOptions 2>&1");
  183. $result = is_string($result) ? explode("\n", $result) : '';
  184. return is_array($result) ? $result[0] : '';
  185. }
  186. }