2013-10-29 12:26:48 +04:00
|
|
|
<?php
|
2017-02-11 18:16:56 +03:00
|
|
|
class InstagramBridge extends BridgeAbstract {
|
|
|
|
|
|
|
|
const MAINTAINER = 'pauder';
|
|
|
|
const NAME = 'Instagram Bridge';
|
2018-10-24 17:33:49 +03:00
|
|
|
const URI = 'https://www.instagram.com/';
|
2017-02-11 18:16:56 +03:00
|
|
|
const DESCRIPTION = 'Returns the newest images';
|
|
|
|
|
2018-03-19 15:29:24 +03:00
|
|
|
const PARAMETERS = array(
|
2019-01-05 14:29:26 +03:00
|
|
|
'Username' => array(
|
2018-03-19 15:29:24 +03:00
|
|
|
'u' => array(
|
|
|
|
'name' => 'username',
|
|
|
|
'required' => true
|
|
|
|
)
|
2017-07-25 16:10:06 +03:00
|
|
|
),
|
2019-01-05 14:29:26 +03:00
|
|
|
'Hashtag' => array(
|
2018-03-19 15:29:24 +03:00
|
|
|
'h' => array(
|
|
|
|
'name' => 'hashtag',
|
|
|
|
'required' => true
|
2018-05-29 13:32:18 +03:00
|
|
|
)
|
|
|
|
),
|
2019-01-05 14:29:26 +03:00
|
|
|
'Location' => array(
|
2018-11-08 00:17:53 +03:00
|
|
|
'l' => array(
|
|
|
|
'name' => 'location',
|
|
|
|
'required' => true
|
|
|
|
)
|
|
|
|
),
|
2018-05-29 13:32:18 +03:00
|
|
|
'global' => array(
|
|
|
|
'media_type' => array(
|
2018-03-19 15:29:24 +03:00
|
|
|
'name' => 'Media type',
|
|
|
|
'type' => 'list',
|
|
|
|
'required' => false,
|
|
|
|
'values' => array(
|
2018-05-29 13:32:18 +03:00
|
|
|
'All' => 'all',
|
2018-03-19 15:29:24 +03:00
|
|
|
'Video' => 'video',
|
2018-05-29 13:32:18 +03:00
|
|
|
'Picture' => 'picture',
|
2019-12-01 13:25:20 +03:00
|
|
|
'Multiple' => 'multiple',
|
2018-03-19 15:29:24 +03:00
|
|
|
),
|
|
|
|
'defaultValue' => 'all'
|
2019-10-03 23:14:21 +03:00
|
|
|
),
|
|
|
|
'direct_links' => array(
|
2019-12-01 13:25:20 +03:00
|
|
|
'name' => 'Use direct media links',
|
2019-10-03 23:14:21 +03:00
|
|
|
'type' => 'checkbox',
|
2018-03-19 15:29:24 +03:00
|
|
|
)
|
2017-02-11 18:16:56 +03:00
|
|
|
)
|
2018-05-29 13:32:18 +03:00
|
|
|
|
2018-03-19 15:29:24 +03:00
|
|
|
);
|
2017-02-11 18:16:56 +03:00
|
|
|
|
2019-09-10 15:37:50 +03:00
|
|
|
const USER_QUERY_HASH = '58b6785bea111c67129decbe6a448951';
|
|
|
|
const TAG_QUERY_HASH = '174a5243287c5f3a7de741089750ab3b';
|
2019-12-01 13:25:20 +03:00
|
|
|
const SHORTCODE_QUERY_HASH = '865589822932d1b43dfe312121dd353a';
|
2019-09-10 15:37:50 +03:00
|
|
|
|
|
|
|
protected function getInstagramUserId($username) {
|
|
|
|
|
|
|
|
if(is_numeric($username)) return $username;
|
|
|
|
|
|
|
|
$cacheFac = new CacheFactory();
|
|
|
|
$cacheFac->setWorkingDir(PATH_LIB_CACHES);
|
|
|
|
$cache = $cacheFac->create(Configuration::getConfig('cache', 'type'));
|
|
|
|
$cache->setScope(get_called_class());
|
2019-11-01 20:06:38 +03:00
|
|
|
$cache->setKey(array($username));
|
2019-09-10 15:37:50 +03:00
|
|
|
$key = $cache->loadData();
|
|
|
|
|
2019-09-11 20:28:46 +03:00
|
|
|
if($key == null) {
|
2019-09-10 15:37:50 +03:00
|
|
|
$data = getContents(self::URI . 'web/search/topsearch/?query=' . $username);
|
2019-09-11 20:08:12 +03:00
|
|
|
|
|
|
|
foreach(json_decode($data)->users as $user) {
|
|
|
|
if($user->user->username === $username) {
|
|
|
|
$key = $user->user->pk;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if($key == null) {
|
|
|
|
returnServerError('Unable to find username in search result.');
|
|
|
|
}
|
2019-09-10 15:37:50 +03:00
|
|
|
$cache->saveData($key);
|
|
|
|
}
|
|
|
|
return $key;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-02-11 18:16:56 +03:00
|
|
|
public function collectData(){
|
2019-10-03 23:14:21 +03:00
|
|
|
$directLink = !is_null($this->getInput('direct_links')) && $this->getInput('direct_links');
|
|
|
|
|
2018-05-05 15:00:59 +03:00
|
|
|
$data = $this->getInstagramJSON($this->getURI());
|
2017-02-11 18:16:56 +03:00
|
|
|
|
2018-03-19 15:29:24 +03:00
|
|
|
if(!is_null($this->getInput('u'))) {
|
2019-09-10 15:37:50 +03:00
|
|
|
$userMedia = $data->data->user->edge_owner_to_timeline_media->edges;
|
2018-11-08 00:17:53 +03:00
|
|
|
} elseif(!is_null($this->getInput('h'))) {
|
2019-09-10 15:37:50 +03:00
|
|
|
$userMedia = $data->data->hashtag->edge_hashtag_to_media->edges;
|
2018-11-08 00:17:53 +03:00
|
|
|
} elseif(!is_null($this->getInput('l'))) {
|
|
|
|
$userMedia = $data->entry_data->LocationsPage[0]->graphql->location->edge_location_to_media->edges;
|
2018-03-19 15:29:24 +03:00
|
|
|
}
|
2017-02-11 18:16:56 +03:00
|
|
|
|
2017-07-29 20:28:00 +03:00
|
|
|
foreach($userMedia as $media) {
|
2018-03-19 15:17:42 +03:00
|
|
|
$media = $media->node;
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
switch($this->getInput('media_type')) {
|
|
|
|
case 'all': break;
|
|
|
|
case 'video':
|
|
|
|
if($media->__typename != 'GraphVideo' || !$media->is_video) continue 2;
|
|
|
|
break;
|
|
|
|
case 'picture':
|
|
|
|
if($media->__typename != 'GraphImage') continue 2;
|
|
|
|
break;
|
|
|
|
case 'multiple':
|
|
|
|
if($media->__typename != 'GraphSidecar') continue 2;
|
|
|
|
break;
|
|
|
|
default: break;
|
2017-07-25 16:10:06 +03:00
|
|
|
}
|
|
|
|
|
2017-02-11 18:16:56 +03:00
|
|
|
$item = array();
|
2018-03-19 15:17:42 +03:00
|
|
|
$item['uri'] = self::URI . 'p/' . $media->shortcode . '/';
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-01-06 20:14:23 +03:00
|
|
|
if (isset($media->owner->username)) {
|
|
|
|
$item['author'] = $media->owner->username;
|
|
|
|
}
|
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
$textContent = $this->getTextContent($media);
|
2018-12-12 18:44:37 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
$item['title'] = ($media->is_video ? '▶ ' : '') . $textContent;
|
2018-12-12 18:44:37 +03:00
|
|
|
$titleLinePos = strpos(wordwrap($item['title'], 120), "\n");
|
|
|
|
if ($titleLinePos != false) {
|
|
|
|
$item['title'] = substr($item['title'], 0, $titleLinePos) . '...';
|
2017-02-11 18:16:56 +03:00
|
|
|
}
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
switch($media->__typename) {
|
|
|
|
case 'GraphSidecar':
|
|
|
|
$data = $this->getInstagramSidecarData($item['uri'], $item['title']);
|
|
|
|
$item['content'] = $data[0];
|
|
|
|
$item['enclosures'] = $data[1];
|
|
|
|
break;
|
|
|
|
case 'GraphImage':
|
|
|
|
if($directLink) {
|
|
|
|
$mediaURI = $media->display_url;
|
|
|
|
} else {
|
|
|
|
$mediaURI = self::URI . 'p/' . $media->shortcode . '/media?size=l';
|
|
|
|
}
|
|
|
|
$item['content'] = '<a href="' . htmlentities($item['uri']) . '" target="_blank">';
|
|
|
|
$item['content'] .= '<img src="' . htmlentities($mediaURI) . '" alt="' . $item['title'] . '" />';
|
|
|
|
$item['content'] .= '</a><br><br>' . nl2br(htmlentities($textContent));
|
|
|
|
$item['enclosures'] = array($mediaURI);
|
|
|
|
break;
|
|
|
|
case 'GraphVideo':
|
|
|
|
$data = $this->getInstagramVideoData($item['uri']);
|
|
|
|
$item['content'] = $data[0];
|
|
|
|
if($directLink) {
|
|
|
|
$item['enclosures'] = $data[1];
|
|
|
|
} else {
|
|
|
|
$item['enclosures'] = array(self::URI . 'p/' . $media->shortcode . '/media?size=l');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: break;
|
2018-05-05 15:00:59 +03:00
|
|
|
}
|
2018-03-19 15:17:42 +03:00
|
|
|
$item['timestamp'] = $media->taken_at_timestamp;
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2017-02-11 18:16:56 +03:00
|
|
|
$this->items[] = $item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
// returns Sidecar(a post which has multiple media)'s contents and enclosures
|
|
|
|
protected function getInstagramSidecarData($uri, $postTitle) {
|
|
|
|
$mediaInfo = $this->getSinglePostData($uri);
|
2019-09-11 20:08:12 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
$textContent = $this->getTextContent($mediaInfo);
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
$enclosures = array();
|
|
|
|
$content = '';
|
|
|
|
foreach($mediaInfo->edge_sidecar_to_children->edges as $singleMedia) {
|
|
|
|
$singleMedia = $singleMedia->node;
|
|
|
|
if($singleMedia->is_video) {
|
|
|
|
if(in_array($singleMedia->video_url, $enclosures)) continue; // check if not added yet
|
|
|
|
$content .= '<video controls><source src="' . $singleMedia->video_url . '" type="video/mp4"></video><br>';
|
|
|
|
array_push($enclosures, $singleMedia->video_url);
|
|
|
|
} else {
|
|
|
|
if(in_array($singleMedia->display_url, $enclosures)) continue; // check if not added yet
|
|
|
|
$content .= '<a href="' . $singleMedia->display_url . '" target="_blank">';
|
|
|
|
$content .= '<img src="' . $singleMedia->display_url . '" alt="' . $postTitle . '" />';
|
|
|
|
$content .= '</a><br>';
|
|
|
|
array_push($enclosures, $singleMedia->display_url);
|
|
|
|
}
|
2018-10-18 17:45:03 +03:00
|
|
|
}
|
2019-12-01 13:25:20 +03:00
|
|
|
$content .= '<br>' . nl2br(htmlentities($textContent));
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
return array($content, $enclosures);
|
|
|
|
}
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
// returns Video post's contents and enclosures
|
|
|
|
protected function getInstagramVideoData($uri) {
|
|
|
|
$mediaInfo = $this->getSinglePostData($uri);
|
|
|
|
|
|
|
|
$textContent = $this->getTextContent($mediaInfo);
|
|
|
|
$content = '<video controls><source src="' . $mediaInfo->video_url . '" type="video/mp4"></video><br>';
|
|
|
|
$content .= '<br>' . nl2br(htmlentities($textContent));
|
|
|
|
|
|
|
|
return array($content, array($mediaInfo->video_url));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getTextContent($media) {
|
|
|
|
$textContent = '(no text)';
|
|
|
|
//Process the first element, that isn't in the node graph
|
|
|
|
if (count($media->edge_media_to_caption->edges) > 0) {
|
|
|
|
$textContent = trim($media->edge_media_to_caption->edges[0]->node->text);
|
2018-05-05 15:00:59 +03:00
|
|
|
}
|
2019-12-01 13:25:20 +03:00
|
|
|
return $textContent;
|
|
|
|
}
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
protected function getSinglePostData($uri) {
|
|
|
|
$shortcode = explode('/', $uri)[4];
|
|
|
|
$data = getContents(self::URI .
|
|
|
|
'graphql/query/?query_hash=' .
|
|
|
|
self::SHORTCODE_QUERY_HASH .
|
|
|
|
'&variables={"shortcode"%3A"' .
|
|
|
|
$shortcode .
|
|
|
|
'"}');
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-12-01 13:25:20 +03:00
|
|
|
return json_decode($data)->data->shortcode_media;
|
2018-05-05 15:00:59 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function getInstagramJSON($uri) {
|
|
|
|
|
2019-09-10 15:37:50 +03:00
|
|
|
if(!is_null($this->getInput('u'))) {
|
|
|
|
|
|
|
|
$userId = $this->getInstagramUserId($this->getInput('u'));
|
|
|
|
|
|
|
|
$data = getContents(self::URI .
|
|
|
|
'graphql/query/?query_hash=' .
|
|
|
|
self::USER_QUERY_HASH .
|
|
|
|
'&variables={"id"%3A"' .
|
|
|
|
$userId .
|
|
|
|
'"%2C"first"%3A10}');
|
|
|
|
return json_decode($data);
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-09-10 15:37:50 +03:00
|
|
|
} elseif(!is_null($this->getInput('h'))) {
|
|
|
|
$data = getContents(self::URI .
|
|
|
|
'graphql/query/?query_hash=' .
|
|
|
|
self::TAG_QUERY_HASH .
|
|
|
|
'&variables={"tag_name"%3A"' .
|
|
|
|
$this->getInput('h') .
|
|
|
|
'"%2C"first"%3A10}');
|
|
|
|
return json_decode($data);
|
|
|
|
|
|
|
|
} else {
|
2018-05-05 15:00:59 +03:00
|
|
|
|
2019-09-10 15:37:50 +03:00
|
|
|
$html = getContents($uri)
|
|
|
|
or returnServerError('Could not request Instagram.');
|
|
|
|
$scriptRegex = '/window\._sharedData = (.*);<\/script>/';
|
|
|
|
|
|
|
|
preg_match($scriptRegex, $html, $matches, PREG_OFFSET_CAPTURE, 0);
|
|
|
|
|
|
|
|
return json_decode($matches[1][0]);
|
|
|
|
|
|
|
|
}
|
2018-05-05 15:00:59 +03:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-02-11 18:16:56 +03:00
|
|
|
public function getName(){
|
2017-07-29 20:28:00 +03:00
|
|
|
if(!is_null($this->getInput('u'))) {
|
2017-02-15 00:20:55 +03:00
|
|
|
return $this->getInput('u') . ' - Instagram Bridge';
|
|
|
|
}
|
|
|
|
|
|
|
|
return parent::getName();
|
2017-02-11 18:16:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getURI(){
|
2017-07-29 20:28:00 +03:00
|
|
|
if(!is_null($this->getInput('u'))) {
|
2018-10-24 17:33:49 +03:00
|
|
|
return self::URI . urlencode($this->getInput('u')) . '/';
|
2018-03-19 15:29:24 +03:00
|
|
|
} elseif(!is_null($this->getInput('h'))) {
|
|
|
|
return self::URI . 'explore/tags/' . urlencode($this->getInput('h'));
|
2018-11-08 00:17:53 +03:00
|
|
|
} elseif(!is_null($this->getInput('l'))) {
|
|
|
|
return self::URI . 'explore/locations/' . urlencode($this->getInput('l'));
|
2017-02-15 00:36:33 +03:00
|
|
|
}
|
|
|
|
return parent::getURI();
|
2017-02-11 18:16:56 +03:00
|
|
|
}
|
2013-10-29 12:26:48 +04:00
|
|
|
}
|