From 41df17bc464832118a767ceff6c9217071699783 Mon Sep 17 00:00:00 2001 From: Dag Date: Sun, 1 Oct 2023 19:23:30 +0200 Subject: [PATCH] refactor (#3712) * test: refactor test suite * docs * refactor * yup * docs --- README.md | 2 + actions/ConnectivityAction.php | 12 - actions/DetectAction.php | 12 - actions/DisplayAction.php | 4 +- actions/ListAction.php | 28 +- actions/SetBridgeCacheAction.php | 12 - bridges/DemoBridge.php | 2 +- lib/ApiAuthenticationMiddleware.php | 12 - lib/AuthenticationMiddleware.php | 12 - lib/BridgeAbstract.php | 162 +++++------ lib/BridgeCard.php | 20 -- lib/BridgeFactory.php | 8 +- lib/Configuration.php | 12 - lib/FeedExpander.php | 12 - lib/FormatFactory.php | 12 - lib/ParameterValidator.php | 272 +++++++----------- lib/bootstrap.php | 14 +- lib/http.php | 6 +- lib/php8backports.php | 34 --- .../BridgeImplementationTest.php | 103 ++++--- tests/CacheImplementationTest.php | 36 +++ tests/Caches/CacheImplementationTest.php | 48 ---- tests/ParameterValidatorTest.php | 40 +++ 23 files changed, 309 insertions(+), 566 deletions(-) rename tests/{Bridges => }/BridgeImplementationTest.php (65%) create mode 100644 tests/CacheImplementationTest.php delete mode 100644 tests/Caches/CacheImplementationTest.php create mode 100644 tests/ParameterValidatorTest.php diff --git a/README.md b/README.md index a1e5fdc7..7037095e 100644 --- a/README.md +++ b/README.md @@ -248,6 +248,8 @@ Modify `report_limit` so that an error must occur 3 times before it is reported. ; Defines how often an error must occur before it is reported to the user report_limit = 3 +The report count is reset to 0 each day. + ### How to password-protect the instance HTTP basic access authentication: diff --git a/actions/ConnectivityAction.php b/actions/ConnectivityAction.php index cfffd195..1568333a 100644 --- a/actions/ConnectivityAction.php +++ b/actions/ConnectivityAction.php @@ -1,17 +1,5 @@ loadConfiguration(); // Remove parameters that don't concern bridges - $bridgeData = array_diff_key($request, array_fill_keys(['action', 'bridge', 'format', '_noproxy', '_cache_timeout', '_error_time'], '')); - $bridge->setDatas($bridgeData); + $input = array_diff_key($request, array_fill_keys(['action', 'bridge', 'format', '_noproxy', '_cache_timeout', '_error_time'], '')); + $bridge->setInput($input); $bridge->collectData(); $items = $bridge->getItems(); if (isset($items[0]) && is_array($items[0])) { diff --git a/actions/ListAction.php b/actions/ListAction.php index 9025bf6e..19bb4d37 100644 --- a/actions/ListAction.php +++ b/actions/ListAction.php @@ -1,17 +1,5 @@ create($bridgeClassName); $list->bridges[$bridgeClassName] = [ - 'status' => $bridgeFactory->isEnabled($bridgeClassName) ? 'active' : 'inactive', - 'uri' => $bridge->getURI(), - 'donationUri' => $bridge->getDonationURI(), - 'name' => $bridge->getName(), - 'icon' => $bridge->getIcon(), - 'parameters' => $bridge->getParameters(), - 'maintainer' => $bridge->getMaintainer(), - 'description' => $bridge->getDescription() + 'status' => $bridgeFactory->isEnabled($bridgeClassName) ? 'active' : 'inactive', + 'uri' => $bridge->getURI(), + 'donationUri' => $bridge->getDonationURI(), + 'name' => $bridge->getName(), + 'icon' => $bridge->getIcon(), + 'parameters' => $bridge->getParameters(), + 'maintainer' => $bridge->getMaintainer(), + 'description' => $bridge->getDescription() ]; } $list->total = count($list->bridges); diff --git a/actions/SetBridgeCacheAction.php b/actions/SetBridgeCacheAction.php index c9264a27..2e9d7147 100644 --- a/actions/SetBridgeCacheAction.php +++ b/actions/SetBridgeCacheAction.php @@ -1,17 +1,5 @@ $optionValue) { + $section = $this->getShortName(); + $configurationOption = Configuration::getConfig($section, $optionName); + + if ($configurationOption !== null) { + $this->configuration[$optionName] = $configurationOption; + continue; + } + + if (isset($optionValue['required']) && $optionValue['required'] === true) { + throw new \Exception(sprintf('Missing configuration option: %s', $optionName)); + } elseif (isset($optionValue['defaultValue'])) { + $this->configuration[$optionName] = $optionValue['defaultValue']; + } + } + } + + public function setInput(array $input) + { + $context = $input['context'] ?? null; + if ($context) { + // Context hinting (optional) + $this->queriedContext = $context; + unset($input['context']); + } + + $parameters = $this->getParameters(); + + if (!$parameters) { + if ($input) { + throw new \Exception('Invalid parameters value(s)'); + } + return; + } + + $validator = new ParameterValidator(); + + // $input is passed by reference! + if (!$validator->validateInput($input, $parameters)) { + $invalidParameterKeys = array_column($validator->getInvalidParameters(), 'name'); + throw new \Exception(sprintf('Invalid parameters value(s): %s', implode(', ', $invalidParameterKeys))); + } + + // Guess the context from input data + if (empty($this->queriedContext)) { + $queriedContext = $validator->getQueriedContext($input, $parameters); + $this->queriedContext = $queriedContext; + } + + if (is_null($this->queriedContext)) { + throw new \Exception('Required parameter(s) missing'); + } elseif ($this->queriedContext === false) { + throw new \Exception('Mixed context parameters'); + } + + $this->setInputWithContext($input, $this->queriedContext); + } + + private function setInputWithContext(array $input, $queriedContext) { // Import and assign all inputs to their context - foreach ($inputs as $name => $value) { + foreach ($input as $name => $value) { foreach (static::PARAMETERS as $context => $set) { if (array_key_exists($name, static::PARAMETERS[$context])) { $this->inputs[$context][$name]['value'] = $value; @@ -128,7 +181,7 @@ abstract class BridgeAbstract switch ($type) { case 'checkbox': - $this->inputs[$context][$name]['value'] = $inputs[$context][$name]['value'] ?? false; + $this->inputs[$context][$name]['value'] = $input[$context][$name]['value'] ?? false; break; case 'list': if (!isset($properties['defaultValue'])) { @@ -153,8 +206,8 @@ abstract class BridgeAbstract // Copy global parameter values to the guessed context if (array_key_exists('global', static::PARAMETERS)) { foreach (static::PARAMETERS['global'] as $name => $properties) { - if (isset($inputs[$name])) { - $value = $inputs[$name]; + if (isset($input[$name])) { + $value = $input[$name]; } else { if ($properties['type'] ?? null === 'checkbox') { $value = false; @@ -176,91 +229,6 @@ abstract class BridgeAbstract } } - /** - * Set inputs for the bridge - * - * Returns errors and aborts execution if the provided input parameters are - * invalid. - * - * @param array List of input parameters. Each element in this list must - * relate to an item in {@see BridgeAbstract::PARAMETERS} - * @return void - */ - public function setDatas(array $inputs) - { - if (isset($inputs['context'])) { // Context hinting (optional) - $this->queriedContext = $inputs['context']; - unset($inputs['context']); - } - - if (empty(static::PARAMETERS)) { - if (!empty($inputs)) { - throw new \Exception('Invalid parameters value(s)'); - } - - return; - } - - $validator = new ParameterValidator(); - - if (!$validator->validateData($inputs, static::PARAMETERS)) { - $parameters = array_map( - function ($i) { - return $i['name']; - }, // Just display parameter names - $validator->getInvalidParameters() - ); - - throw new \Exception(sprintf('Invalid parameters value(s): %s', implode(', ', $parameters))); - } - - // Guess the context from input data - if (empty($this->queriedContext)) { - $this->queriedContext = $validator->getQueriedContext($inputs, static::PARAMETERS); - } - - if (is_null($this->queriedContext)) { - throw new \Exception('Required parameter(s) missing'); - } elseif ($this->queriedContext === false) { - throw new \Exception('Mixed context parameters'); - } - - $this->setInputs($inputs, $this->queriedContext); - } - - /** - * Loads configuration for the bridge - * - * Returns errors and aborts execution if the provided configuration is - * invalid. - * - * @return void - */ - public function loadConfiguration() - { - foreach (static::CONFIGURATION as $optionName => $optionValue) { - $section = $this->getShortName(); - $configurationOption = Configuration::getConfig($section, $optionName); - - if ($configurationOption !== null) { - $this->configuration[$optionName] = $configurationOption; - continue; - } - - if (isset($optionValue['required']) && $optionValue['required'] === true) { - throw new \Exception(sprintf('Missing configuration option: %s', $optionName)); - } elseif (isset($optionValue['defaultValue'])) { - $this->configuration[$optionName] = $optionValue['defaultValue']; - } - } - } - - /** - * Returns the value for the provided input - * - * @param string $input The input name - * @return mixed|null The input value or null if the input is not defined - */ protected function getInput($input) { return $this->inputs[$this->queriedContext][$input]['value'] ?? null; diff --git a/lib/BridgeCard.php b/lib/BridgeCard.php index 99c44fff..b2fda192 100644 --- a/lib/BridgeCard.php +++ b/lib/BridgeCard.php @@ -1,25 +1,5 @@ invalid[] = [ - 'name' => $name, - 'reason' => $reason, - ]; - } - - /** - * Return list of invalid parameters. - * - * Each element is an array of 'name' and 'reason'. - * - * @return array List of invalid parameters - */ - public function getInvalidParameters() - { - return $this->invalid; - } - - /** - * Validate value for a text input - * - * @param string $value The value of a text input - * @param string|null $pattern (optional) A regex pattern - * @return string|null The filtered value or null if the value is invalid - */ - private function validateTextValue($value, $pattern = null) - { - if (!is_null($pattern)) { - $filteredValue = filter_var( - $value, - FILTER_VALIDATE_REGEXP, - ['options' => [ - 'regexp' => '/^' . $pattern . '$/' - ] - ] - ); - } else { - $filteredValue = filter_var($value); - } - - if ($filteredValue === false) { - return null; - } - - return $filteredValue; - } - - /** - * Validate value for a number input - * - * @param int $value The value of a number input - * @return int|null The filtered value or null if the value is invalid - */ - private function validateNumberValue($value) - { - $filteredValue = filter_var($value, FILTER_VALIDATE_INT); - - if ($filteredValue === false) { - return null; - } - - return $filteredValue; - } - - /** - * Validate value for a checkbox - * - * @param bool $value The value of a checkbox - * @return bool The filtered value - */ - private function validateCheckboxValue($value) - { - return filter_var($value, FILTER_VALIDATE_BOOLEAN, FILTER_NULL_ON_FAILURE); - } - - /** - * Validate value for a list - * - * @param string $value The value of a list - * @param array $expectedValues A list of expected values - * @return string|null The filtered value or null if the value is invalid - */ - private function validateListValue($value, $expectedValues) - { - $filteredValue = filter_var($value); - - if ($filteredValue === false) { - return null; - } - - if (!in_array($filteredValue, $expectedValues)) { // Check sub-values? - foreach ($expectedValues as $subName => $subValue) { - if (is_array($subValue) && in_array($filteredValue, $subValue)) { - return $filteredValue; - } - } - return null; - } - - return $filteredValue; - } - - /** - * Check if all required parameters are satisfied - * - * @param array $data (ref) A list of input values - * @param array $parameters The bridge parameters - * @return bool True if all parameters are satisfied - */ - public function validateData(&$data, $parameters) - { - if (!is_array($data)) { + if (!is_array($input)) { return false; } - foreach ($data as $name => $value) { + foreach ($input as $name => $value) { // Some RSS readers add a cache-busting parameter (_=) to feed URLs, detect and ignore them. if ($name === '_') { continue; @@ -156,54 +23,60 @@ class ParameterValidator $registered = false; foreach ($parameters as $context => $set) { - if (array_key_exists($name, $set)) { - $registered = true; - if (!isset($set[$name]['type'])) { - $set[$name]['type'] = 'text'; - } + if (!array_key_exists($name, $set)) { + continue; + } + $registered = true; + if (!isset($set[$name]['type'])) { + // Default type is text + $set[$name]['type'] = 'text'; + } - switch ($set[$name]['type']) { - case 'number': - $data[$name] = $this->validateNumberValue($value); - break; - case 'checkbox': - $data[$name] = $this->validateCheckboxValue($value); - break; - case 'list': - $data[$name] = $this->validateListValue($value, $set[$name]['values']); - break; - default: - case 'text': - if (isset($set[$name]['pattern'])) { - $data[$name] = $this->validateTextValue($value, $set[$name]['pattern']); - } else { - $data[$name] = $this->validateTextValue($value); - } - break; - } + switch ($set[$name]['type']) { + case 'number': + $input[$name] = $this->validateNumberValue($value); + break; + case 'checkbox': + $input[$name] = $this->validateCheckboxValue($value); + break; + case 'list': + $input[$name] = $this->validateListValue($value, $set[$name]['values']); + break; + default: + case 'text': + if (isset($set[$name]['pattern'])) { + $input[$name] = $this->validateTextValue($value, $set[$name]['pattern']); + } else { + $input[$name] = $this->validateTextValue($value); + } + break; + } - if (is_null($data[$name]) && isset($set[$name]['required']) && $set[$name]['required']) { - $this->addInvalidParameter($name, 'Parameter is invalid!'); - } + if ( + is_null($input[$name]) + && isset($set[$name]['required']) + && $set[$name]['required'] + ) { + $this->invalid[] = ['name' => $name, 'reason' => 'Parameter is invalid!']; } } if (!$registered) { - $this->addInvalidParameter($name, 'Parameter is not registered!'); + $this->invalid[] = ['name' => $name, 'reason' => 'Parameter is not registered!']; } } - return empty($this->invalid); + return $this->invalid === []; } /** * Get the name of the context matching the provided inputs * - * @param array $data Associative array of user data + * @param array $input Associative array of user data * @param array $parameters Array of bridge parameters * @return string|null Returns the context name or null if no match was found */ - public function getQueriedContext($data, $parameters) + public function getQueriedContext($input, $parameters) { $queriedContexts = []; @@ -212,7 +85,7 @@ class ParameterValidator $queriedContexts[$context] = null; // Ensure all user data exist in the current context - $notInContext = array_diff_key($data, $set); + $notInContext = array_diff_key($input, $set); if (array_key_exists('global', $parameters)) { $notInContext = array_diff_key($notInContext, $parameters['global']); } @@ -222,7 +95,7 @@ class ParameterValidator // Check if all parameters of the context are satisfied foreach ($set as $id => $properties) { - if (isset($data[$id]) && !empty($data[$id])) { + if (isset($input[$id]) && !empty($input[$id])) { $queriedContexts[$context] = true; } elseif ( isset($properties['type']) @@ -248,8 +121,8 @@ class ParameterValidator switch (array_sum($queriedContexts)) { case 0: // Found no match, is there a context without parameters? - if (isset($data['context'])) { - return $data['context']; + if (isset($input['context'])) { + return $input['context']; } foreach ($queriedContexts as $context => $queried) { if (is_null($queried)) { @@ -264,4 +137,55 @@ class ParameterValidator return false; } } + + public function getInvalidParameters(): array + { + return $this->invalid; + } + + private function validateTextValue($value, $pattern = null) + { + if (is_null($pattern)) { + // No filtering taking place + $filteredValue = filter_var($value); + } else { + $filteredValue = filter_var($value, FILTER_VALIDATE_REGEXP, ['options' => ['regexp' => '/^' . $pattern . '$/']]); + } + if ($filteredValue === false) { + return null; + } + return $filteredValue; + } + + private function validateNumberValue($value) + { + $filteredValue = filter_var($value, FILTER_VALIDATE_INT); + if ($filteredValue === false) { + return null; + } + return $filteredValue; + } + + private function validateCheckboxValue($value) + { + return filter_var($value, FILTER_VALIDATE_BOOLEAN, FILTER_NULL_ON_FAILURE); + } + + private function validateListValue($value, $expectedValues) + { + $filteredValue = filter_var($value); + if ($filteredValue === false) { + return null; + } + if (!in_array($filteredValue, $expectedValues)) { + // Check sub-values? + foreach ($expectedValues as $subName => $subValue) { + if (is_array($subValue) && in_array($filteredValue, $subValue)) { + return $filteredValue; + } + } + return null; + } + return $filteredValue; + } } diff --git a/lib/bootstrap.php b/lib/bootstrap.php index bc584541..a95de9dd 100644 --- a/lib/bootstrap.php +++ b/lib/bootstrap.php @@ -1,18 +1,6 @@ body; } - public function getCode() + public function getCode(): int { return $this->code; } @@ -226,7 +226,7 @@ final class Response return self::STATUS_CODES[$this->code] ?? ''; } - public function getHeaders() + public function getHeaders(): array { return $this->headers; } diff --git a/lib/php8backports.php b/lib/php8backports.php index 5b103e3d..ccef6016 100644 --- a/lib/php8backports.php +++ b/lib/php8backports.php @@ -1,39 +1,5 @@ setBridge($path); - $this->assertTrue($this->class === ucfirst($this->class), 'class name must start with uppercase character'); - $this->assertEquals(0, substr_count($this->class, ' '), 'class name must not contain spaces'); - $this->assertStringEndsWith('Bridge', $this->class, 'class name must end with "Bridge"'); + $this->assertTrue($this->className === ucfirst($this->className), 'class name must start with uppercase character'); + $this->assertEquals(0, substr_count($this->className, ' '), 'class name must not contain spaces'); + $this->assertStringEndsWith('Bridge', $this->className, 'class name must end with "Bridge"'); } /** @@ -28,7 +28,7 @@ class BridgeImplementationTest extends TestCase public function testClassType($path) { $this->setBridge($path); - $this->assertInstanceOf(BridgeAbstract::class, $this->obj); + $this->assertInstanceOf(BridgeAbstract::class, $this->bridge); } /** @@ -38,18 +38,18 @@ class BridgeImplementationTest extends TestCase { $this->setBridge($path); - $this->assertIsString($this->obj::NAME, 'class::NAME'); - $this->assertNotEmpty($this->obj::NAME, 'class::NAME'); - $this->assertIsString($this->obj::URI, 'class::URI'); - $this->assertNotEmpty($this->obj::URI, 'class::URI'); - $this->assertIsString($this->obj::DESCRIPTION, 'class::DESCRIPTION'); - $this->assertNotEmpty($this->obj::DESCRIPTION, 'class::DESCRIPTION'); - $this->assertIsString($this->obj::MAINTAINER, 'class::MAINTAINER'); - $this->assertNotEmpty($this->obj::MAINTAINER, 'class::MAINTAINER'); + $this->assertIsString($this->bridge::NAME, 'class::NAME'); + $this->assertNotEmpty($this->bridge::NAME, 'class::NAME'); + $this->assertIsString($this->bridge::URI, 'class::URI'); + $this->assertNotEmpty($this->bridge::URI, 'class::URI'); + $this->assertIsString($this->bridge::DESCRIPTION, 'class::DESCRIPTION'); + $this->assertNotEmpty($this->bridge::DESCRIPTION, 'class::DESCRIPTION'); + $this->assertIsString($this->bridge::MAINTAINER, 'class::MAINTAINER'); + $this->assertNotEmpty($this->bridge::MAINTAINER, 'class::MAINTAINER'); - $this->assertIsArray($this->obj::PARAMETERS, 'class::PARAMETERS'); - $this->assertIsInt($this->obj::CACHE_TIMEOUT, 'class::CACHE_TIMEOUT'); - $this->assertGreaterThanOrEqual(0, $this->obj::CACHE_TIMEOUT, 'class::CACHE_TIMEOUT'); + $this->assertIsArray($this->bridge::PARAMETERS, 'class::PARAMETERS'); + $this->assertIsInt($this->bridge::CACHE_TIMEOUT, 'class::CACHE_TIMEOUT'); + $this->assertGreaterThanOrEqual(0, $this->bridge::CACHE_TIMEOUT, 'class::CACHE_TIMEOUT'); } /** @@ -60,23 +60,22 @@ class BridgeImplementationTest extends TestCase $this->setBridge($path); $multiMinimum = 2; - if (isset($this->obj::PARAMETERS['global'])) { + if (isset($this->bridge::PARAMETERS['global'])) { ++$multiMinimum; } - $multiContexts = (count($this->obj::PARAMETERS) >= $multiMinimum); + $multiContexts = (count($this->bridge::PARAMETERS) >= $multiMinimum); $paramsSeen = []; $allowedTypes = [ 'text', 'number', 'list', - 'checkbox' + 'checkbox', ]; - foreach ($this->obj::PARAMETERS as $context => $params) { + foreach ($this->bridge::PARAMETERS as $context => $params) { if ($multiContexts) { $this->assertIsString($context, 'invalid context name'); - $this->assertNotEmpty($context, 'The context name cannot be empty'); } @@ -152,11 +151,10 @@ class BridgeImplementationTest extends TestCase } } - foreach ($this->obj::TEST_DETECT_PARAMETERS as $url => $params) { - $this->assertEquals($this->obj->detectParameters($url), $params); + foreach ($this->bridge::TEST_DETECT_PARAMETERS as $url => $params) { + $detectedParameters = $this->bridge->detectParameters($url); + $this->assertEquals($detectedParameters, $params); } - - $this->assertTrue(true); } /** @@ -164,19 +162,21 @@ class BridgeImplementationTest extends TestCase */ public function testVisibleMethods($path) { - $allowedBridgeAbstract = get_class_methods(BridgeAbstract::class); - sort($allowedBridgeAbstract); - $allowedFeedExpander = get_class_methods(FeedExpander::class); - sort($allowedFeedExpander); + $bridgeAbstractMethods = get_class_methods(BridgeAbstract::class); + sort($bridgeAbstractMethods); + $feedExpanderMethods = get_class_methods(FeedExpander::class); + sort($feedExpanderMethods); $this->setBridge($path); - $methods = get_class_methods($this->obj); - sort($methods); - if ($this->obj instanceof FeedExpander) { - $this->assertEquals($allowedFeedExpander, $methods); - } else { - $this->assertEquals($allowedBridgeAbstract, $methods); + $publicMethods = get_class_methods($this->bridge); + sort($publicMethods); + foreach ($publicMethods as $publicMethod) { + if ($this->bridge instanceof FeedExpander) { + $this->assertContains($publicMethod, $feedExpanderMethods); + } else { + $this->assertContains($publicMethod, $bridgeAbstractMethods); + } } } @@ -187,23 +187,23 @@ class BridgeImplementationTest extends TestCase { $this->setBridge($path); - $value = $this->obj->getDescription(); + $value = $this->bridge->getDescription(); $this->assertIsString($value, '$class->getDescription()'); $this->assertNotEmpty($value, '$class->getDescription()'); - $value = $this->obj->getMaintainer(); + $value = $this->bridge->getMaintainer(); $this->assertIsString($value, '$class->getMaintainer()'); $this->assertNotEmpty($value, '$class->getMaintainer()'); - $value = $this->obj->getName(); + $value = $this->bridge->getName(); $this->assertIsString($value, '$class->getName()'); $this->assertNotEmpty($value, '$class->getName()'); - $value = $this->obj->getURI(); + $value = $this->bridge->getURI(); $this->assertIsString($value, '$class->getURI()'); $this->assertNotEmpty($value, '$class->getURI()'); - $value = $this->obj->getIcon(); + $value = $this->bridge->getIcon(); $this->assertIsString($value, '$class->getIcon()'); } @@ -214,14 +214,14 @@ class BridgeImplementationTest extends TestCase { $this->setBridge($path); - $this->checkUrl($this->obj::URI); - $this->checkUrl($this->obj->getURI()); + $this->assertNotFalse(filter_var($this->bridge::URI, FILTER_VALIDATE_URL)); + $this->assertNotFalse(filter_var($this->bridge->getURI(), FILTER_VALIDATE_URL)); } public function dataBridgesProvider() { $bridges = []; - foreach (glob(__DIR__ . '/../../bridges/*Bridge.php') as $path) { + foreach (glob(__DIR__ . '/../bridges/*Bridge.php') as $path) { $bridges[basename($path, '.php')] = [$path]; } return $bridges; @@ -229,16 +229,11 @@ class BridgeImplementationTest extends TestCase private function setBridge($path) { - $this->class = '\\' . basename($path, '.php'); - $this->assertTrue(class_exists($this->class), 'class ' . $this->class . ' doesn\'t exist'); - $this->obj = new $this->class( + $this->className = '\\' . basename($path, '.php'); + $this->assertTrue(class_exists($this->className), 'class ' . $this->className . ' doesn\'t exist'); + $this->bridge = new $this->className( new \NullCache(), - new \NullLogger() + new \NullLogger(), ); } - - private function checkUrl($url) - { - $this->assertNotFalse(filter_var($url, FILTER_VALIDATE_URL), 'no valid URL: ' . $url); - } } diff --git a/tests/CacheImplementationTest.php b/tests/CacheImplementationTest.php new file mode 100644 index 00000000..e6ed352b --- /dev/null +++ b/tests/CacheImplementationTest.php @@ -0,0 +1,36 @@ +assertTrue($path === ucfirst($path), 'class name must start with uppercase character'); + $this->assertEquals(0, substr_count($path, ' '), 'class name must not contain spaces'); + $this->assertStringEndsWith('Cache', $path, 'class name must end with "Cache"'); + } + + /** + * @dataProvider getCacheClassNames + */ + public function testClassType($path) + { + $this->assertTrue(is_subclass_of($path, CacheInterface::class), 'class must be subclass of CacheInterface'); + } +} diff --git a/tests/Caches/CacheImplementationTest.php b/tests/Caches/CacheImplementationTest.php deleted file mode 100644 index a3ad5f79..00000000 --- a/tests/Caches/CacheImplementationTest.php +++ /dev/null @@ -1,48 +0,0 @@ -setCache($path); - $this->assertTrue($this->class === ucfirst($this->class), 'class name must start with uppercase character'); - $this->assertEquals(0, substr_count($this->class, ' '), 'class name must not contain spaces'); - $this->assertStringEndsWith('Cache', $this->class, 'class name must end with "Cache"'); - } - - /** - * @dataProvider dataCachesProvider - */ - public function testClassType($path) - { - $this->setCache($path); - $this->assertTrue(is_subclass_of($this->class, CacheInterface::class), 'class must be subclass of CacheInterface'); - } - - //////////////////////////////////////////////////////////////////////////// - - public function dataCachesProvider() - { - $caches = []; - foreach (glob(PATH_LIB_CACHES . '*.php') as $path) { - $caches[basename($path, '.php')] = [$path]; - } - return $caches; - } - - private function setCache($path) - { - $this->class = '\\' . basename($path, '.php'); - $this->assertTrue(class_exists($this->class), 'class ' . $this->class . ' doesn\'t exist'); - } -} diff --git a/tests/ParameterValidatorTest.php b/tests/ParameterValidatorTest.php new file mode 100644 index 00000000..1b241c2c --- /dev/null +++ b/tests/ParameterValidatorTest.php @@ -0,0 +1,40 @@ + 'joe']; + $parameters = [ + [ + 'user' => [ + 'name' => 'User', + 'type' => 'text', + ], + ] + ]; + $this->assertTrue($sut->validateInput($input, $parameters)); + } + + public function test2() + { + $sut = new \ParameterValidator(); + $input = ['username' => 'joe']; + $parameters = [ + [ + 'user' => [ + 'name' => 'User', + 'type' => 'text', + ], + ] + ]; + $this->assertFalse($sut->validateInput($input, $parameters)); + } +}