 Compolomus    /
                    RssReader
                      Compolomus    /
                    RssReader
                
                            | 1 | <?php declare(strict_types=1); | ||
| 2 | |||
| 3 | namespace Compolomus\RssReader; | ||
| 4 | |||
| 5 | use DateTime; | ||
| 6 | use DOMDocument; | ||
| 7 | use DOMXPath; | ||
| 8 | use function array_column; | ||
| 9 | use function array_merge; | ||
| 10 | use function count; | ||
| 11 | use function crc32; | ||
| 12 | use function in_array; | ||
| 13 | use function strip_tags; | ||
| 14 | use function trim; | ||
| 15 | |||
| 16 | class RssReader | ||
| 17 | { | ||
| 18 | public function __construct( | ||
| 19 | public array $channels = [], | ||
| 20 | public ?CacheInterface $cache = null, | ||
| 21 | public int $limit = 0, | ||
| 22 |     ) { | ||
| 23 |         if (!empty($_ENV['RSSREADER_LIMIT'])) { | ||
| 24 | $this->limit = (int) $_ENV['RSSREADER_LIMIT']; | ||
| 25 | } | ||
| 26 | |||
| 27 |         if (null === $this->cache) { | ||
| 28 | $this->cache = new FileCache(); | ||
| 29 | } | ||
| 30 | } | ||
| 31 | |||
| 32 | /** | ||
| 33 | * @throws \Exception | ||
| 34 | */ | ||
| 35 | protected function getPostsFromChannel(string $chanel): ?array | ||
| 36 |     { | ||
| 37 | // Load document | ||
| 38 | $dom = new DOMDocument(); | ||
| 39 | $dom->load($chanel); | ||
| 40 | |||
| 41 | // Extract all items | ||
| 42 |         $items = $dom->getElementsByTagName('item'); | ||
| 43 | |||
| 44 | // Build XPath object | ||
| 45 | $xpath = new DOMXpath($dom); | ||
| 46 | |||
| 47 | // Parse all items in loop | ||
| 48 | $result = []; | ||
| 49 |         foreach ($items as $item) { | ||
| 50 |             $link      = $item->getElementsByTagName('link')->item(0)->nodeValue; | ||
| 51 | $itemId = crc32($link); | ||
| 52 |             $timestamp = (new DateTime($item->getElementsByTagName('pubDate')->item(0)->nodeValue))->getTimestamp(); | ||
| 53 | |||
| 54 | // Extract posts which is not in a cache | ||
| 55 |             if (!in_array($itemId, $this->cache->getIds(), false)) { | ||
| 0 ignored issues–
                            show | |||
| 56 | $result[] = [ | ||
| 57 | 'id' => $itemId, | ||
| 58 |                     'title'     => $item->getElementsByTagName('title')->item(0)->nodeValue, | ||
| 59 |                     'desc'      => trim(strip_tags($item->getElementsByTagName('description')->item(0)->nodeValue)), | ||
| 60 | 'link' => $link, | ||
| 61 | 'timestamp' => $timestamp, | ||
| 62 |                     'img'       => $xpath->query('//enclosure/@url')->item(0)->nodeValue, | ||
| 63 | ]; | ||
| 64 | } | ||
| 65 | } | ||
| 66 | |||
| 67 | return $result; | ||
| 68 | } | ||
| 69 | |||
| 70 | /** | ||
| 71 | * Return array with only unique elements | ||
| 72 | * | ||
| 73 | * @param array $array | ||
| 74 | * | ||
| 75 | * @return array | ||
| 76 | */ | ||
| 77 | public function getUnique(array $array): array | ||
| 78 |     { | ||
| 79 |         $serialized = array_map('serialize', $array); | ||
| 80 | $unique = array_unique($serialized); | ||
| 81 | |||
| 82 | return array_intersect_key($array, $unique); | ||
| 83 | } | ||
| 84 | |||
| 85 | /** | ||
| 86 | * @throws \Exception | ||
| 87 | */ | ||
| 88 | public function getAll(): array | ||
| 89 |     { | ||
| 90 | $result = []; | ||
| 91 |         foreach ($this->channels as $chanel) { | ||
| 92 | $result[] = $this->getPostsFromChannel($chanel); | ||
| 93 | } | ||
| 94 | |||
| 95 | // Merge results from all sources | ||
| 96 | $result = array_merge(...$result); | ||
| 97 | |||
| 98 | // Only unique records | ||
| 99 | $result = $this->getUnique($result); | ||
| 100 | |||
| 101 | // Sort array by timestamp column | ||
| 102 |         usort($result, static function ($a, $b) { | ||
| 103 |             if ($a['timestamp'] === $b['timestamp']) { | ||
| 104 | return 0; | ||
| 105 | } | ||
| 106 | // ASC order | ||
| 107 | return ($a['timestamp'] > $b['timestamp']) ? 1 : -1; | ||
| 108 | }); | ||
| 109 | |||
| 110 | // Extract IDs and save them all | ||
| 111 | $ids = array_column($result, 'id'); | ||
| 112 |         if (count($ids)) { | ||
| 113 | $this->cache->saveIds($ids); | ||
| 114 | } | ||
| 115 | |||
| 116 | // Slice last few posts if limit is set | ||
| 117 |         if (!empty($this->limit)) { | ||
| 118 | $result = array_slice($result, -$this->limit, $this->limit); | ||
| 119 | } | ||
| 120 | |||
| 121 | return $result; | ||
| 122 | } | ||
| 123 | } | ||
| 124 | 
 
                                
This check looks for calls to methods that do not seem to exist on a given type. It looks for the method on the type itself as well as in inherited classes or implemented interfaces.
This is most likely a typographical error or the method has been renamed.