From e6aef73a02f7220f2a2ce8acc9d4f6c627ceb88a Mon Sep 17 00:00:00 2001 From: Dag Date: Wed, 20 Sep 2023 02:45:48 +0200 Subject: [PATCH] refactor (#3668) --- actions/DisplayAction.php | 8 +- docs/05_Bridge_API/02_BridgeAbstract.md | 49 ++--- docs/05_Bridge_API/03_FeedExpander.md | 5 +- docs/05_Bridge_API/index.md | 6 +- docs/07_Cache_API/index.md | 8 +- docs/09_Technical_recommendations/index.md | 2 +- lib/BridgeAbstract.php | 213 +++++---------------- lib/BridgeCard.php | 6 +- lib/BridgeFactory.php | 2 +- lib/BridgeInterface.php | 145 -------------- lib/FeedExpander.php | 2 +- lib/RssBridge.php | 21 +- lib/contents.php | 31 ++- lib/http.php | 17 +- tests/Bridges/BridgeImplementationTest.php | 3 +- 15 files changed, 134 insertions(+), 384 deletions(-) delete mode 100644 lib/BridgeInterface.php diff --git a/actions/DisplayAction.php b/actions/DisplayAction.php index 7c59b3d5..efa4d3b5 100644 --- a/actions/DisplayAction.php +++ b/actions/DisplayAction.php @@ -84,7 +84,7 @@ class DisplayAction implements ActionInterface return $response; } - private function createResponse(array $request, BridgeInterface $bridge, FormatInterface $format) + private function createResponse(array $request, BridgeAbstract $bridge, FormatInterface $format) { $items = []; $infos = []; @@ -110,8 +110,6 @@ class DisplayAction implements ActionInterface 'icon' => $bridge->getIcon() ]; } catch (\Exception $e) { - $errorOutput = Configuration::getConfig('error', 'output'); - $reportLimit = Configuration::getConfig('error', 'report_limit'); if ($e instanceof HttpException) { // Reproduce (and log) these responses regardless of error output and report limit if ($e->getCode() === 429) { @@ -124,6 +122,8 @@ class DisplayAction implements ActionInterface } } Logger::error(sprintf('Exception in DisplayAction(%s)', $bridge->getShortName()), ['e' => $e]); + $errorOutput = Configuration::getConfig('error', 'output'); + $reportLimit = Configuration::getConfig('error', 'report_limit'); $errorCount = 1; if ($reportLimit > 1) { $errorCount = $this->logBridgeError($bridge->getName(), $e->getCode()); @@ -152,7 +152,7 @@ class DisplayAction implements ActionInterface return new Response($format->stringify(), 200, $headers); } - private function createFeedItemFromException($e, BridgeInterface $bridge): FeedItem + private function createFeedItemFromException($e, BridgeAbstract $bridge): FeedItem { $item = new FeedItem(); diff --git a/docs/05_Bridge_API/02_BridgeAbstract.md b/docs/05_Bridge_API/02_BridgeAbstract.md index 12d24bdc..a8e9db42 100644 --- a/docs/05_Bridge_API/02_BridgeAbstract.md +++ b/docs/05_Bridge_API/02_BridgeAbstract.md @@ -94,7 +94,7 @@ class MyBridge extends BridgeAbstract { const MAINTAINER = 'ghost'; public function collectData() { - $item = array(); // Create an empty item + $item = []; // Create an empty item $item['title'] = 'Hello World!'; @@ -121,11 +121,11 @@ class MyBridge extends BridgeAbstract { const URI = ''; const DESCRIPTION = 'No description provided'; const MAINTAINER = 'No maintainer'; - const PARAMETERS = array(); // Can be omitted! + const PARAMETERS = []; // Can be omitted! const CACHE_TIMEOUT = 3600; // Can be omitted! public function collectData() { - $item = array(); // Create an empty item + $item = []; // Create an empty item $item['title'] = 'Hello World!'; @@ -145,7 +145,7 @@ For information on how to read parameter values during execution, please refer t ## Adding parameters to a bridge -Parameters are specified as part of the bridge class. An empty list of parameters is defined as `const PARAMETERS = array();` +Parameters are specified as part of the bridge class. An empty list of parameters is defined as `const PARAMETERS = [];`
Show example
@@ -153,7 +153,7 @@ Parameters are specified as part of the bridge class. An empty list of parameter Show example
```PHP -const PARAMETERS = array( - 'My Context 1' => array(), - 'My Context 2' => array() -); +const PARAMETERS = [ + 'My Context 1' => [], + 'My Context 2' => [], +]; ``` **Output** @@ -189,9 +189,9 @@ _Notice_: The name of a context can be left empty if only one context is needed!
Show example
```PHP -const PARAMETERS = array( - array() -); +const PARAMETERS = [ + [] +]; ```

@@ -201,25 +201,28 @@ You can also define a set of parameters that will be applied to every possible c
Show example
```PHP -const PARAMETERS = array( - 'global' => array() // Applies to all contexts! -); +const PARAMETERS = [ + 'global' => [] // Applies to all contexts! +]; ```
## Level 2 - Parameter -Parameters are placed inside a context. They are defined as associative array of parameter specifications. Each parameter is defined by it's internal input name, a definition in the form `'n' => array();`, where `n` is the name with which the bridge can access the parameter during execution. +Parameters are placed inside a context. +They are defined as associative array of parameter specifications. +Each parameter is defined by it's internal input name, a definition in the form `'n' => [];`, +where `n` is the name with which the bridge can access the parameter during execution.
Show example
```PHP -const PARAMETERS = array( - 'My Context' => array( - 'n' => array() - ) -); +const PARAMETERS = [ + 'My Context' => [ + 'n' => [] + ] +]; ```

@@ -351,7 +354,7 @@ Elements collected by this function must be stored in `$this->items`. The `items ```PHP -$item = array(); // Create a new item +$item = []; // Create a new item $item['title'] = 'Hello World!'; @@ -448,7 +451,7 @@ public function detectParameters($url){ && preg_match($regex, $url, $urlMatches) > 0 && preg_match($regex, static::URI, $bridgeUriMatches) > 0 && $urlMatches[3] === $bridgeUriMatches[3]) { - return array(); + return []; } else { return null; } diff --git a/docs/05_Bridge_API/03_FeedExpander.md b/docs/05_Bridge_API/03_FeedExpander.md index 7e72670a..910d1abb 100644 --- a/docs/05_Bridge_API/03_FeedExpander.md +++ b/docs/05_Bridge_API/03_FeedExpander.md @@ -93,10 +93,11 @@ class MySiteBridge extends FeedExpander { const NAME = 'Unnamed bridge'; const URI = ''; const DESCRIPTION = 'No description provided'; - const PARAMETERS = array(); + const PARAMETERS = []; const CACHE_TIMEOUT = 3600; - public function collectData(){ + public function collectData() + { $this->collectExpandableDatas('your feed URI'); } } diff --git a/docs/05_Bridge_API/index.md b/docs/05_Bridge_API/index.md index 6115fa01..e49e47be 100644 --- a/docs/05_Bridge_API/index.md +++ b/docs/05_Bridge_API/index.md @@ -1,4 +1,8 @@ -A _Bridge_ is an class that allows **RSS-Bridge** to create an RSS-feed from a website. A _Bridge_ represents one element on the [Welcome screen](../01_General/04_Screenshots.md) and covers one or more sites to return feeds for. It is developed in a PHP file located in the `bridges/` folder (see [Folder structure](../04_For_Developers/03_Folder_structure.md)) and extends one of the base classes of **RSS-Bridge**: +A _Bridge_ is a class that allows **RSS-Bridge** to create an RSS-feed from a website. +A _Bridge_ represents one element on the [Welcome screen](../01_General/04_Screenshots.md) +and covers one or more sites to return feeds for. +It is developed in a PHP file located in the `bridges/` folder (see [Folder structure](../04_For_Developers/03_Folder_structure.md)) +and extends one of the base classes of **RSS-Bridge**: Base class | Description -----------|------------ diff --git a/docs/07_Cache_API/index.md b/docs/07_Cache_API/index.md index 57692847..17afbacc 100644 --- a/docs/07_Cache_API/index.md +++ b/docs/07_Cache_API/index.md @@ -1,4 +1,6 @@ -A _Cache_ is a class that allows **RSS-Bridge** to store fetched data in a local storage area on the server. Cache imlementations are placed in the `caches/` folder (see [Folder structure](../04_For_Developers/03_Folder_structure.md)). A cache must implement the [`CacheInterface`](../07_Cache_API/02_CacheInterface.md) interface. - -For more information about how to create a new `Cache`, read [How to create a new cache?](../07_Cache_API/01_How_to_create_a_new_cache.md) +A _Cache_ is a class that allows **RSS-Bridge** to store fetched data in a local storage area on the server. +Cache imlementations are placed in the `caches/` folder (see [Folder structure](../04_For_Developers/03_Folder_structure.md)). +A cache must implement the [`CacheInterface`](../07_Cache_API/02_CacheInterface.md) interface. +For more information about how to create a new `Cache`, read +[How to create a new cache?](../07_Cache_API/01_How_to_create_a_new_cache.md) diff --git a/docs/09_Technical_recommendations/index.md b/docs/09_Technical_recommendations/index.md index f2b15476..a57f0bbd 100644 --- a/docs/09_Technical_recommendations/index.md +++ b/docs/09_Technical_recommendations/index.md @@ -15,7 +15,7 @@ class TestBridge extends BridgeAbstract { const URI = ''; const DESCRIPTION = 'No description provided'; const MAINTAINER = 'No maintainer'; - const PARAMETERS = array(); + const PARAMETERS = []; const CACHE_TIMEOUT = 3600; public function collectData(){ diff --git a/lib/BridgeAbstract.php b/lib/BridgeAbstract.php index a69552fc..f51fe893 100644 --- a/lib/BridgeAbstract.php +++ b/lib/BridgeAbstract.php @@ -1,76 +1,15 @@ 'Maximum number of items to return', ]; - /** - * Holds the list of items collected by the bridge - * - * Items must be collected by {@see BridgeInterface::collectData()} - * - * Use {@see BridgeAbstract::getItems()} to access items. - * - * @var array - */ protected array $items = []; - - /** - * Holds the list of input parameters used by the bridge - * - * Do not access this parameter directly! - * Use {@see BridgeAbstract::setInputs()} and {@see BridgeAbstract::getInput()} instead! - * - * @var array - */ protected array $inputs = []; - - /** - * Holds the name of the queried context - * - * @var string - */ - protected $queriedContext = ''; - - /** - * Holds the list of bridge-specific configurations from config.ini.php, used by the bridge. - */ + protected string $queriedContext = ''; private array $configuration = []; public function __construct() { } - /** {@inheritdoc} */ + abstract public function collectData(); + public function getItems() { return $this->items; } + public function getOption(string $name) + { + return $this->configuration[$name] ?? null; + } + + public function getDescription() + { + return static::DESCRIPTION; + } + + public function getMaintainer(): string + { + return static::MAINTAINER; + } + + public function getName() + { + return static::NAME; + } + + public function getIcon() + { + return static::URI . '/favicon.ico'; + } + + public function getParameters(): array + { + return static::PARAMETERS; + } + + public function getURI() + { + return static::URI; + } + + public function getDonationURI(): string + { + return static::DONATION_URI; + } + + public function getCacheTimeout() + { + return static::CACHE_TIMEOUT; + } + /** * Sets the input values for a given context. * @@ -299,10 +256,7 @@ abstract class BridgeAbstract implements BridgeInterface */ protected function getInput($input) { - if (!isset($this->inputs[$this->queriedContext][$input]['value'])) { - return null; - } - return $this->inputs[$this->queriedContext][$input]['value']; + return $this->inputs[$this->queriedContext][$input]['value'] ?? null; } /** @@ -340,63 +294,6 @@ abstract class BridgeAbstract implements BridgeInterface } } - /** - * Get bridge configuration value - */ - public function getOption($name) - { - return $this->configuration[$name] ?? null; - } - - /** {@inheritdoc} */ - public function getDescription() - { - return static::DESCRIPTION; - } - - /** {@inheritdoc} */ - public function getMaintainer() - { - return static::MAINTAINER; - } - - /** {@inheritdoc} */ - public function getName() - { - return static::NAME; - } - - /** {@inheritdoc} */ - public function getIcon() - { - return static::URI . '/favicon.ico'; - } - - /** {@inheritdoc} */ - public function getParameters() - { - return static::PARAMETERS; - } - - /** {@inheritdoc} */ - public function getURI() - { - return static::URI; - } - - /** {@inheritdoc} */ - public function getDonationURI() - { - return static::DONATION_URI; - } - - /** {@inheritdoc} */ - public function getCacheTimeout() - { - return static::CACHE_TIMEOUT; - } - - /** {@inheritdoc} */ public function detectParameters($url) { $regex = '/^(https?:\/\/)?(www\.)?(.+?)(\/)?$/'; @@ -411,11 +308,6 @@ abstract class BridgeAbstract implements BridgeInterface return null; } - /** - * Loads a cached value for the specified key - * - * @return mixed Cached value or null if the key doesn't exist or has expired - */ protected function loadCacheValue(string $key) { $cache = RssBridge::getCache(); @@ -423,11 +315,6 @@ abstract class BridgeAbstract implements BridgeInterface return $cache->get($cacheKey); } - /** - * Stores a value to cache with the specified key - * - * @param mixed $value Value to cache - */ protected function saveCacheValue(string $key, $value, $ttl = 86400) { $cache = RssBridge::getCache(); diff --git a/lib/BridgeCard.php b/lib/BridgeCard.php index 6eef3879..99c44fff 100644 --- a/lib/BridgeCard.php +++ b/lib/BridgeCard.php @@ -25,7 +25,7 @@ final class BridgeCard /** * Gets a single bridge card * - * @param class-string $bridgeClassName The bridge name + * @param class-string $bridgeClassName The bridge name * @param array $formats A list of formats * @param bool $isActive Indicates if the bridge is active or not * @return string The bridge card @@ -116,7 +116,7 @@ CARD; /** * Get the form header for a bridge card * - * @param class-string $bridgeClassName The bridge name + * @param class-string $bridgeClassName The bridge name * @param bool $isHttps If disabled, adds a warning to the form * @return string The form header */ @@ -143,7 +143,7 @@ This bridge is not fetching its content through a secure connection
'; /** * Get the form body for a bridge * - * @param class-string $bridgeClassName The bridge name + * @param class-string $bridgeClassName The bridge name * @param array $formats A list of supported formats * @param bool $isActive Indicates if a bridge is enabled or not * @param bool $isHttps Indicates if a bridge uses HTTPS or not diff --git a/lib/BridgeFactory.php b/lib/BridgeFactory.php index f302a27a..12565d92 100644 --- a/lib/BridgeFactory.php +++ b/lib/BridgeFactory.php @@ -34,7 +34,7 @@ final class BridgeFactory } } - public function create(string $name): BridgeInterface + public function create(string $name): BridgeAbstract { return new $name(); } diff --git a/lib/BridgeInterface.php b/lib/BridgeInterface.php deleted file mode 100644 index 63bc7b70..00000000 --- a/lib/BridgeInterface.php +++ /dev/null @@ -1,145 +0,0 @@ - $e]); http_response_code(500); @@ -57,9 +60,6 @@ final class RssBridge } }); - // Consider: ini_set('error_reporting', E_ALL & ~E_DEPRECATED); - date_default_timezone_set(Configuration::getConfig('system', 'timezone')); - self::$httpClient = new CurlHttpClient(); $cacheFactory = new CacheFactory(); @@ -68,11 +68,6 @@ final class RssBridge } else { self::$cache = $cacheFactory->create(); } - - if (Configuration::getConfig('authentication', 'enable')) { - $authenticationMiddleware = new AuthenticationMiddleware(); - $authenticationMiddleware(); - } } public function main(array $argv = []): void @@ -81,6 +76,10 @@ final class RssBridge parse_str(implode('&', array_slice($argv, 1)), $cliArgs); $request = $cliArgs; } else { + if (Configuration::getConfig('authentication', 'enable')) { + $authenticationMiddleware = new AuthenticationMiddleware(); + $authenticationMiddleware(); + } $request = array_merge($_GET, $_POST); } @@ -124,10 +123,4 @@ final class RssBridge { return self::$cache ?? new NullCache(); } - - public function clearCache() - { - $cache = self::getCache(); - $cache->clear(); - } } diff --git a/lib/contents.php b/lib/contents.php index c1847758..e173b542 100644 --- a/lib/contents.php +++ b/lib/contents.php @@ -16,6 +16,13 @@ function getContents( ) { $httpClient = RssBridge::getHttpClient(); + $httpHeadersNormalized = []; + foreach ($httpHeaders as $httpHeader) { + $parts = explode(':', $httpHeader); + $headerName = trim($parts[0]); + $headerValue = trim(implode(':', array_slice($parts, 1))); + $httpHeadersNormalized[$headerName] = $headerValue; + } // Snagged from https://github.com/lwthiker/curl-impersonate/blob/main/firefox/curl_ff102 $defaultHttpHeaders = [ 'Accept' => 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8', @@ -27,13 +34,6 @@ function getContents( 'Sec-Fetch-User' => '?1', 'TE' => 'trailers', ]; - $httpHeadersNormalized = []; - foreach ($httpHeaders as $httpHeader) { - $parts = explode(':', $httpHeader); - $headerName = trim($parts[0]); - $headerValue = trim(implode(':', array_slice($parts, 1))); - $httpHeadersNormalized[$headerName] = $headerValue; - } $config = [ 'useragent' => Configuration::getConfig('http', 'useragent'), 'timeout' => Configuration::getConfig('http', 'timeout'), @@ -43,7 +43,7 @@ function getContents( $maxFileSize = Configuration::getConfig('http', 'max_filesize'); if ($maxFileSize) { - // Multiply with 2^20 (1M) to the value in bytes + // Convert from MB to B by multiplying with 2^20 (1M) $config['max_filesize'] = $maxFileSize * 2 ** 20; } @@ -57,7 +57,6 @@ function getContents( /** @var Response $cachedResponse */ $cachedResponse = $cache->get($cacheKey); if ($cachedResponse) { - // considering popping $cachedLastModified = $cachedResponse->getHeader('last-modified'); if ($cachedLastModified) { $cachedLastModified = new \DateTimeImmutable($cachedLastModified); @@ -101,21 +100,13 @@ function getContents( Debug::isEnabled() ? mb_substr($response->getBody(), 0, 500) : '', ); - // The following code must be extracted if it grows too much - $cloudflareTitles = [ - 'Just a moment...', - '<title>Please Wait...', - '<title>Attention Required!', - '<title>Security | Glassdoor', - ]; - foreach ($cloudflareTitles as $cloudflareTitle) { - if (str_contains($response->getBody(), $cloudflareTitle)) { - throw new CloudFlareException($exceptionMessage, $response->getCode()); - } + if (CloudFlareException::isCloudFlareResponse($response)) { + throw new CloudFlareException($exceptionMessage, $response->getCode()); } throw new HttpException(trim($exceptionMessage), $response->getCode()); } if ($returnFull === true) { + // todo: return the actual response object return [ 'code' => $response->getCode(), 'headers' => $response->getHeaders(), diff --git a/lib/http.php b/lib/http.php index c5e65e77..cc1d0e22 100644 --- a/lib/http.php +++ b/lib/http.php @@ -6,6 +6,21 @@ class HttpException extends \Exception final class CloudFlareException extends HttpException { + public static function isCloudFlareResponse(Response $response): bool + { + $cloudflareTitles = [ + '<title>Just a moment...', + '<title>Please Wait...', + '<title>Attention Required!', + '<title>Security | Glassdoor', + ]; + foreach ($cloudflareTitles as $cloudflareTitle) { + if (str_contains($response->getBody(), $cloudflareTitle)) { + return true; + } + } + return false; + } } interface HttpClient @@ -119,7 +134,7 @@ final class CurlHttpClient implements HttpClient } } - $statusCode = curl_getinfo($ch, CURLINFO_HTTP_CODE); + $statusCode = curl_getinfo($ch, CURLINFO_RESPONSE_CODE); curl_close($ch); return new Response($data, $statusCode, $responseHeaders); } diff --git a/tests/Bridges/BridgeImplementationTest.php b/tests/Bridges/BridgeImplementationTest.php index cf03f356..807649fc 100644 --- a/tests/Bridges/BridgeImplementationTest.php +++ b/tests/Bridges/BridgeImplementationTest.php @@ -3,7 +3,6 @@ namespace RssBridge\Tests\Bridges; use BridgeAbstract; -use BridgeInterface; use FeedExpander; use PHPUnit\Framework\TestCase; @@ -29,7 +28,7 @@ class BridgeImplementationTest extends TestCase public function testClassType($path) { $this->setBridge($path); - $this->assertInstanceOf(BridgeInterface::class, $this->obj); + $this->assertInstanceOf(BridgeAbstract::class, $this->obj); } /**