diff --git a/bridges/DealabsBridge.php b/bridges/DealabsBridge.php index da342501..6a2d902e 100644 --- a/bridges/DealabsBridge.php +++ b/bridges/DealabsBridge.php @@ -1874,12 +1874,30 @@ class DealabsBridge extends PepperBridgeAbstract { 'type' => 'list', 'title' => 'Ordre de tri des deals', 'values' => array( - 'Du deal le plus Hot au moins Hot' => '', + 'Du deal le plus Hot au moins Hot' => '-hot', 'Du deal le plus récent au plus ancien' => '-nouveaux', - 'Du deal le plus commentés au moins commentés' => '-commentes' ) ) - ) + ), + 'Surveillance Discussion' => array( + 'url' => array( + 'name' => 'URL de la discussion', + 'type' => 'text', + 'required' => true, + 'title' => 'URL discussion à surveiller: https://www.dealabs.com/discussions/titre-1234', + 'exampleValue' => 'https://www.dealabs.com/discussions/titre-1234', + ), + + 'only_with_url' => array( + 'name' => 'Exclure les commentaires sans URL', + 'type' => 'checkbox', + 'title' => 'Exclure les commentaires ne contenant pas d\'URL dans le flux', + 'defaultValue' => false, + ) + + + ) + ); public $lang = array( @@ -1887,8 +1905,10 @@ class DealabsBridge extends PepperBridgeAbstract { 'bridge-name' => SELF::NAME, 'context-keyword' => 'Recherche par Mot(s) clé(s)', 'context-group' => 'Deals par groupe', + 'context-talk' => 'Surveillance Discussion', 'uri-group' => 'groupe/', - 'request-error' => 'Could not request Dealabs', + 'request-error' => 'Impossible de joindre Dealabs', + 'thread-error' => 'Impossible de déterminer l\'ID de la discussion. Vérifiez l\'URL que vous avez entré', 'no-results' => 'Il n'y a rien à afficher pour le moment :(', 'relative-date-indicator' => array( 'il y a', @@ -1899,6 +1919,7 @@ class DealabsBridge extends PepperBridgeAbstract { 'discount' => 'Réduction', 'title-keyword' => 'Recherche', 'title-group' => 'Groupe', + 'title-talk' => 'Surveillance Discussion', 'local-months' => array( 'janvier', 'février', @@ -1954,6 +1975,9 @@ class PepperBridgeAbstract extends BridgeAbstract { case $this->i8n('context-group'): return $this->collectDataGroup(); break; + case $this->i8n('context-talk'): + return $this->collectDataTalk(); + break; } } @@ -2039,20 +2063,16 @@ class PepperBridgeAbstract extends BridgeAbstract { } else { foreach ($list as $deal) { $item = array(); - $item['uri'] = $deal->find('div[class*=threadGrid-title]', 0)->find('a', 0)->href; - $item['title'] = $deal->find('a[class*=' . $selectorLink . ']', 0 - )->plaintext; + $item['uri'] = $this->getDealURI($deal); + $item['title'] = $this->GetTitle($deal); $item['author'] = $deal->find('span.thread-username', 0)->plaintext; + $item['content'] = '

' - . $deal->find('a[class*=' . $selectorLink . ']', 0)->innertext - . '

' + . '"/>
' + . $this->getHTMLTitle($item) . $this->getPrice($deal) . $this->getDiscount($deal) . $this->getShipsFrom($deal) @@ -2080,6 +2100,127 @@ class PepperBridgeAbstract extends BridgeAbstract { } } + /** + * Get the Talk lastest comments + */ + protected function collectDataTalk(){ + $threadURL = $this->getInput('url'); + $onlyWithUrl = $this->getInput('only_with_url'); + + // Get Thread ID from url passed in parameter + $threadSearch = preg_match('/-([0-9]{1,20})$/', $threadURL, $matches); + + // Show an error message if we can't find the thread ID in the URL sent by the user + if($threadSearch !== 1) { + returnClientError($this->i8n('thread-error')); + } + $threadID = $matches[1]; + + $url = $this->i8n('bridge-uri') . 'graphql'; + + // Get Cookies header to do the query + $cookies = $this->getCookies($url); + + // GraphQL String + // This was extracted from https://www.dealabs.com/assets/js/modern/common_211b99.js + // This string was extracted during a Website visit, and minified using this neat tool : + // https://codepen.io/dangodev/pen/Baoqmoy + $graphqlString = <<<'HEREDOC' +query comments($filter:CommentFilter!,$limit:Int,$page:Int){comments(filter:$filter,limit:$limit,page:$page){ +items{...commentFields}pagination{...paginationFields}}}fragment commentFields on Comment{commentId threadId url +preparedHtmlContent user{...userMediumAvatarFields...userNameFields...userPersonaFields bestBadge{...badgeFields}} +reactionCounts{type count}deletable currentUserReaction{type}reported reportable source status createdAt updatedAt +ignored popular deletedBy{username}notes{content createdAt user{username}}lastEdit{reason timeAgo userId}}fragment +userMediumAvatarFields on User{userId isDeletedOrPendingDeletion imageUrls(slot:"default",variations: +["user_small_avatar"])}fragment userNameFields on User{userId username isUserProfileHidden isDeletedOrPendingDeletion} +fragment userPersonaFields on User{persona{type text}}fragment badgeFields on Badge{badgeId level{...badgeLevelFields}} +fragment badgeLevelFields on BadgeLevel{key name description}fragment paginationFields on Pagination{count current last + next previous size order} +HEREDOC; + + // Construct the JSON object to send to the Website + $queryArray = array ( + 'query' => $graphqlString, + 'variables' => array ( + 'filter' => array ( + 'threadId' => array ( + 'eq' => $threadID, + ), + 'order' => array ( + 'direction' => 'Descending', + ), + + ), + 'page' => 1, + ), + ); + $queryJSON = json_encode($queryArray); + + // HTTP headers + $header = array( + 'Content-Type: application/json', + 'Accept: application/json, text/plain, */*', + 'X-Pepper-Txn: threads.show', + 'X-Request-Type: application/vnd.pepper.v1+json', + 'X-Requested-With: XMLHttpRequest', + $cookies, + ); + // CURL Options + $opts = array( + CURLOPT_POST => 1, + CURLOPT_POSTFIELDS => $queryJSON + ); + $json = getContents($url, $header, $opts); + $objects = json_decode($json); + foreach($objects->data->comments->items as $comment) { + $item = array(); + $item['uri'] = $comment->url; + $item['title'] = $comment->user->username . ' - ' . $comment->createdAt; + $item['author'] = $comment->user->username; + $item['content'] = $comment->preparedHtmlContent; + $item['uid'] = $comment->commentId; + // Timestamp handling needs a new parsing function + if($onlyWithUrl == true) { + // Count Links and Quote Links + $content = str_get_html($item['content']); + $countLinks = count($content->find('a[href]')); + $countQuoteLinks = count($content->find('a[href][class=userHtml-quote-source]')); + // Only add element if there are Links ans more links tant Quote links + if($countLinks > 0 && $countLinks > $countQuoteLinks) { + $this->items[] = $item; + } + } else { + $this->items[] = $item; + } + } + } + + /** + * Extract the cookies obtained from the URL + * @return array the array containing the cookies set by the URL + */ + private function getCookies($url) + { + $ch = curl_init($url); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); + // get headers too with this line + curl_setopt($ch, CURLOPT_HEADER, 1); + $result = curl_exec($ch); + // get cookie + // multi-cookie variant contributed by @Combuster in comments + preg_match_all('/^Set-Cookie:\s*([^;]*)/mi', $result, $matches); + $cookies = array(); + foreach($matches[1] as $item) { + parse_str($item, $cookie); + $cookies = array_merge($cookies, $cookie); + } + $header = 'Cookie: '; + foreach($cookies as $name => $content) { + $header .= $name . '=' . $content . '; '; + } + return $header; + } + /** * Check if the string $str contains any of the string of the array $arr * @return boolean true if the string matched anything otherwise false @@ -2112,6 +2253,74 @@ class PepperBridgeAbstract extends BridgeAbstract { } } + /** + * Get the Title from a Deal if it exists + * @return string String of the deal title + */ + private function getTitle($deal) + { + + $titleRoot = $deal->find('div[class*=threadGrid-title]', 0); + $titleA = $titleRoot->find('a[class*=thread-link]', 0); + $titleFirstChild = $titleRoot->first_child(); + if($titleA !== null) { + $title = $titleA->plaintext; + } else { + // Inb ssome case, expired deals have a different format + $title = $titleRoot->find('span', 0)->plaintext; + } + + return $title; + + } + + /** + * Get the Title from a Talk if it exists + * @return string String of the Talk title + */ + private function getTalkTitle() + { + $html = getSimpleHTMLDOMCached($this->getInput('url')); + $title = $html->find('h1[class=thread-title]', 0)->plaintext; + return $title; + + } + + /** + * Get the HTML Title code from an item + * @return string String of the deal title + */ + private function getHTMLTitle($item) + { + if($item['uri'] == '') { + $html = '

' . $item['title'] . '

'; + } else { + $html = '

' + . $item['title'] . '

'; + } + + return $html; + + } + + /** + * Get the URI from a Deal if it exists + * @return string String of the deal URI + */ + private function getDealURI($deal) + { + + $uriA = $deal->find('div[class*=threadGrid-title]', 0)->find('a[class*=thread-link]', 0); + if($uriA === null) { + $uri = ''; + } else { + $uri = $uriA->href; + } + + return $uri; + + } + /** * Get the Shipping costs from a Deal if it exists * @return string String of the deal shipping Cost @@ -2342,13 +2551,16 @@ class PepperBridgeAbstract extends BridgeAbstract { $group = array_search($this->getInput('group'), $values); return $this->i8n('bridge-name') . ' - ' . $this->i8n('title-group') . ' : ' . $group; break; + case $this->i8n('context-talk'): + return $this->i8n('bridge-name') . ' - ' . $this->i8n('title-talk') . ' : ' . $this->getTalkTitle(); + break; default: // Return default value return static::NAME; } } /** - * Returns the RSS Feed title according to the parameters + * Returns the RSS Feed URI according to the parameters * @return string the RSS feed Title */ public function getURI(){ @@ -2359,6 +2571,9 @@ class PepperBridgeAbstract extends BridgeAbstract { case $this->i8n('context-group'): return $this->getGroupURI(); break; + case $this->i8n('context-talk'): + return $this->getTalkURI(); + break; default: // Return default value return static::URI; } @@ -2373,7 +2588,7 @@ class PepperBridgeAbstract extends BridgeAbstract { $hide_expired = $this->getInput('hide_expired'); $hide_local = $this->getInput('hide_local'); $priceFrom = $this->getInput('priceFrom'); - $priceTo = $this->getInput('priceFrom'); + $priceTo = $this->getInput('priceTo'); $url = $this->i8n('bridge-uri') . 'search/advanced?q=' . urlencode($q) @@ -2403,6 +2618,15 @@ class PepperBridgeAbstract extends BridgeAbstract { return $url; } + /** + * Returns the RSS Feed URI for a Talk Feed + * @return string the RSS feed URI + */ + private function getTalkURI(){ + $url = $this->getInput('url'); + return $url; + } + /** * This is some "localisation" function that returns the needed content using * the "$lang" class variable in the local class diff --git a/bridges/HotUKDealsBridge.php b/bridges/HotUKDealsBridge.php index 60a1ba25..08fcd75c 100644 --- a/bridges/HotUKDealsBridge.php +++ b/bridges/HotUKDealsBridge.php @@ -3243,11 +3243,27 @@ class HotUKDealsBridge extends PepperBridgeAbstract { 'title' => 'Sort order of deals', 'values' => array( 'From the most to the least hot deal' => '-hot', - 'From the most recent deal to the oldest' => '', - 'From the most commented deal to the least commented deal' => '-discussed' + 'From the most recent deal to the oldest' => '-new', ) ) - ) + ), + 'Discussion Monitoring' => array( + 'url' => array( + 'name' => 'Discussion URL', + 'type' => 'text', + 'required' => true, + 'title' => 'Discussion URL to monitor. Ex: https://www.hotukdeals.com/discussions/the-hukd-lego-thread-3599357', + 'exampleValue' => 'https://www.hotukdeals.com/discussions/the-hukd-lego-thread-3599357', + ), + 'only_with_url' => array( + 'name' => 'Exclude comments without URL', + 'type' => 'checkbox', + 'title' => 'Exclude comments that does not contains URL in the feed', + 'defaultValue' => false, + ) + ) + + ); public $lang = array( @@ -3255,8 +3271,10 @@ class HotUKDealsBridge extends PepperBridgeAbstract { 'bridge-name' => SELF::NAME, 'context-keyword' => 'Search by keyword(s))', 'context-group' => 'Deals per group', + 'context-talk' => 'Discussion Monitoring', 'uri-group' => 'tag/', 'request-error' => 'Could not request HotUKDeals', + 'thread-error' => 'Unable to determine the thread ID. Check the URL you entered', 'no-results' => 'Ooops, looks like we could', 'relative-date-indicator' => array( 'ago', @@ -3267,6 +3285,7 @@ class HotUKDealsBridge extends PepperBridgeAbstract { 'discount' => 'Discount', 'title-keyword' => 'Search', 'title-group' => 'Group', + 'title-talk' => 'Discussion Monitoring', 'local-months' => array( 'Jan', 'Feb', diff --git a/bridges/MydealsBridge.php b/bridges/MydealsBridge.php index 1bcc444f..ebff8159 100644 --- a/bridges/MydealsBridge.php +++ b/bridges/MydealsBridge.php @@ -1991,12 +1991,26 @@ class MydealsBridge extends PepperBridgeAbstract { 'type' => 'list', 'title' => 'Sortierung der deals', 'values' => array( - 'Vom heißesten zum kältesten Deal' => '', + 'Vom heißesten zum kältesten Deal' => '-hot', 'Vom jüngsten Deal zum ältesten' => '-new', - 'Vom am meisten kommentierten Deal zum am wenigsten kommentierten Deal' => '-discussed' + ) + ), + ), + 'Überwachung Diskussion' => array( + 'url' => array( + 'name' => 'URL der Diskussion', + 'type' => 'text', + 'required' => true, + 'title' => 'URL-Diskussion zu überwachen: https://www.mydealz.de/diskussion/title-123', + 'exampleValue' => '://www.mydealz.de/diskussion/title-123', + ), + 'only_with_url' => array( + 'name' => 'Kommentare ohne URL ausschließen', + 'type' => 'checkbox', + 'title' => 'Kommentare, die keine URL enthalten, im Feed ausschließen', + 'defaultValue' => false, ) ) - ) ); public $lang = array( @@ -2004,8 +2018,10 @@ class MydealsBridge extends PepperBridgeAbstract { 'bridge-name' => SELF::NAME, 'context-keyword' => 'Suche nach Stichworten', 'context-group' => 'Deals pro Gruppen', + 'context-talk' => 'Überwachung Diskussion', 'uri-group' => 'gruppe/', 'request-error' => 'Could not request mydeals', + 'thread-error' => 'Die ID der Diskussion kann nicht ermittelt werden. Überprüfen Sie die eingegebene URL', 'no-results' => 'Ups, wir konnten keine Deals zu', 'relative-date-indicator' => array( 'vor', @@ -2017,6 +2033,7 @@ class MydealsBridge extends PepperBridgeAbstract { 'discount' => 'Rabatte', 'title-keyword' => 'Suche', 'title-group' => 'Gruppe', + 'title-talk' => 'Überwachung Diskussion', 'local-months' => array( 'Jan', 'Feb',