From 97f07cf2168bc7b72a67d72e8ac7205de8cbbc30 Mon Sep 17 00:00:00 2001 From: sysadminstory Date: Wed, 5 Mar 2025 19:32:03 +0100 Subject: [PATCH] [InstagramBridge] Add a fallback to the "Username" mode (#4461) - Added some header that could help Instagram to not block RSS Bridge - Added a fallback function to use the "Embed profile" Instagram feature to get the content shared by one Instagram user --- bridges/InstagramBridge.php | 45 +++++++++++++++++++++++++++++++++---- 1 file changed, 41 insertions(+), 4 deletions(-) diff --git a/bridges/InstagramBridge.php b/bridges/InstagramBridge.php index 633d6080..d16b925c 100644 --- a/bridges/InstagramBridge.php +++ b/bridges/InstagramBridge.php @@ -86,6 +86,11 @@ class InstagramBridge extends BridgeAbstract $headers = []; $sessionId = $this->getOption('session_id'); $dsUserId = $this->getOption('ds_user_id'); + $headers[] = 'x-ig-app-id: 936619743392459'; + $headers[] = 'User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/62.0.3202.94 Safari/537.36'; + $headers[] = 'Accept-Language: en-US,en;q=0.9,ru;q=0.8'; + $headers[] = 'Accept-Encoding: gzip, deflate, br'; + $headers[] = 'Accept: */*'; if ($sessionId and $dsUserId) { $headers[] = 'cookie: sessionid=' . $sessionId . '; ds_user_id=' . $dsUserId; } @@ -125,8 +130,10 @@ class InstagramBridge extends BridgeAbstract return; } - if (!is_null($this->getInput('u'))) { + if (!is_null($this->getInput('u')) && !$this->fallbackMode) { $userMedia = $data->data->user->edge_owner_to_timeline_media->edges; + } elseif (!is_null($this->getInput('u')) && $this->fallbackMode) { + $userMedia = $data->context->graphql_media; } elseif (!is_null($this->getInput('h'))) { $userMedia = $data->data->hashtag->edge_hashtag_to_media->edges; } elseif (!is_null($this->getInput('l'))) { @@ -134,7 +141,12 @@ class InstagramBridge extends BridgeAbstract } foreach ($userMedia as $media) { - $media = $media->node; + // The media is not in the same element if in fallback mode than not + if (!$this->fallbackMode) { + $media = $media->node; + } else { + $media = $media->shortcode_media; + } switch ($this->getInput('media_type')) { case 'all': @@ -267,14 +279,39 @@ class InstagramBridge extends BridgeAbstract protected function getInstagramJSON($uri) { + // Sets fallbackMode to false + $this->fallbackMode = false; if (!is_null($this->getInput('u'))) { - $userId = $this->getInstagramUserId($this->getInput('u')); - $data = $this->getContents(self::URI . + try { + $userId = $this->getInstagramUserId($this->getInput('u')); + $data = $this->getContents(self::URI . 'graphql/query/?query_hash=' . self::USER_QUERY_HASH . '&variables={"id"%3A"' . $userId . '"%2C"first"%3A10}'); + } catch (HttpException $e) { + // If loading the data directly failed, we fall back to the "/embed" data loading + // We are in the fallback mode : set a booolean to handle this specific case while collecting the content + $this->fallbackMode = true; + // Get the HTML code of the profile embed page, and extract the JSON of it + $username = $this->getInput('u'); + // Load the content using the integrated function to use helping headers + $htmlString = $this->getContents(self::URI . $username . '/embed/'); + // Load the String as an SimpleHTMLDom Object + $html = new simple_html_dom(); + $html->load($htmlString); + // Find the