2013-08-15 12:05:58 +02:00
|
|
|
<?php
|
|
|
|
/**
|
2016-07-08 19:06:35 +02:00
|
|
|
* RssBridgeYoutube
|
2013-08-15 12:05:58 +02:00
|
|
|
* Returns the newest videos
|
2015-11-01 12:17:36 +01:00
|
|
|
* WARNING: to parse big playlists (over ~90 videos), you need to edit simple_html_dom.php:
|
2014-05-14 14:34:06 +02:00
|
|
|
* change: define('MAX_FILE_SIZE', 600000);
|
|
|
|
* into: define('MAX_FILE_SIZE', 900000); (or more)
|
2013-08-15 12:05:58 +02:00
|
|
|
*/
|
2015-11-01 12:17:36 +01:00
|
|
|
class YoutubeBridge extends BridgeAbstract {
|
|
|
|
|
2016-08-30 11:23:55 +02:00
|
|
|
const NAME = 'YouTube Bridge';
|
|
|
|
const URI = 'https://www.youtube.com/';
|
2016-09-25 17:04:28 +02:00
|
|
|
const CACHE_TIMEOUT = 10800; // 3h
|
2016-08-30 11:23:55 +02:00
|
|
|
const DESCRIPTION = 'Returns the 10 newest videos by username/channel/playlist or search';
|
|
|
|
const MAINTAINER = 'mitsukarenai';
|
2015-11-01 12:17:36 +01:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
const PARAMETERS = array(
|
|
|
|
'By username' => array(
|
|
|
|
'u' => array(
|
|
|
|
'name' => 'username',
|
|
|
|
'exampleValue' => 'test',
|
|
|
|
'required' => true
|
|
|
|
)
|
|
|
|
),
|
|
|
|
'By channel id' => array(
|
|
|
|
'c' => array(
|
|
|
|
'name' => 'channel id',
|
2018-06-29 23:55:33 +02:00
|
|
|
'exampleValue' => '15',
|
2017-02-11 16:16:56 +01:00
|
|
|
'required' => true
|
|
|
|
)
|
|
|
|
),
|
|
|
|
'By playlist Id' => array(
|
|
|
|
'p' => array(
|
|
|
|
'name' => 'playlist id',
|
2018-06-29 23:55:33 +02:00
|
|
|
'exampleValue' => '15'
|
2017-02-11 16:16:56 +01:00
|
|
|
)
|
|
|
|
),
|
|
|
|
'Search result' => array(
|
|
|
|
's' => array(
|
|
|
|
'name' => 'search keyword',
|
|
|
|
'exampleValue' => 'test'
|
|
|
|
),
|
|
|
|
'pa' => array(
|
|
|
|
'name' => 'page',
|
|
|
|
'type' => 'number',
|
|
|
|
'exampleValue' => 1
|
|
|
|
)
|
2018-07-21 14:22:53 +02:00
|
|
|
),
|
|
|
|
'global' => array(
|
|
|
|
'duration_min' => array(
|
|
|
|
'name' => 'min. duration (minutes)',
|
|
|
|
'type' => 'number',
|
|
|
|
'title' => 'Minimum duration for the video in minutes',
|
|
|
|
'exampleValue' => 5
|
|
|
|
),
|
|
|
|
'duration_max' => array(
|
|
|
|
'name' => 'max. duration (minutes)',
|
|
|
|
'type' => 'number',
|
|
|
|
'title' => 'Maximum duration for the video in minutes',
|
|
|
|
'exampleValue' => 10
|
|
|
|
)
|
2017-02-11 16:16:56 +01:00
|
|
|
)
|
|
|
|
);
|
2015-11-01 12:17:36 +01:00
|
|
|
|
2018-07-21 14:22:53 +02:00
|
|
|
private $feedName = '';
|
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
private function ytBridgeQueryVideoInfo($vid, &$author, &$desc, &$time){
|
2017-04-27 21:40:20 +02:00
|
|
|
$html = $this->ytGetSimpleHTMLDOM(self::URI . "watch?v=$vid");
|
2018-02-16 22:11:03 +01:00
|
|
|
|
|
|
|
// Skip unavailable videos
|
2018-02-16 22:35:00 +01:00
|
|
|
if(!strpos($html->innertext, 'IS_UNAVAILABLE_PAGE')) {
|
2018-02-16 22:11:03 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-02-16 22:35:00 +01:00
|
|
|
foreach($html->find('script') as $script) {
|
2018-02-16 22:12:24 +01:00
|
|
|
$data = trim($script->innertext);
|
|
|
|
|
|
|
|
if(strpos($data, '{') !== 0)
|
|
|
|
continue; // Wrong script
|
|
|
|
|
|
|
|
$json = json_decode($data);
|
|
|
|
|
|
|
|
if(!isset($json->itemListElement))
|
|
|
|
continue; // Wrong script
|
|
|
|
|
|
|
|
$author = $json->itemListElement[0]->item->name;
|
|
|
|
}
|
2017-08-17 19:26:01 +02:00
|
|
|
|
2018-02-16 22:26:18 +01:00
|
|
|
if(!is_null($html->find('#watch-description-text', 0)))
|
|
|
|
$desc = $html->find('#watch-description-text', 0)->innertext;
|
2017-08-17 19:26:01 +02:00
|
|
|
|
|
|
|
if(!is_null($html->find('meta[itemprop=datePublished]', 0)))
|
|
|
|
$time = strtotime($html->find('meta[itemprop=datePublished]', 0)->getAttribute('content'));
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
2014-06-20 16:41:51 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
private function ytBridgeAddItem($vid, $title, $author, $desc, $time){
|
2016-08-22 18:55:59 +02:00
|
|
|
$item = array();
|
|
|
|
$item['id'] = $vid;
|
|
|
|
$item['title'] = $title;
|
|
|
|
$item['author'] = $author;
|
|
|
|
$item['timestamp'] = $time;
|
2017-02-11 16:16:56 +01:00
|
|
|
$item['uri'] = self::URI . 'watch?v=' . $vid;
|
|
|
|
$thumbnailUri = str_replace('/www.', '/img.', self::URI) . 'vi/' . $vid . '/0.jpg';
|
|
|
|
$item['content'] = '<a href="' . $item['uri'] . '"><img src="' . $thumbnailUri . '" /></a><br />' . $desc;
|
2016-05-03 23:45:32 +02:00
|
|
|
$this->items[] = $item;
|
|
|
|
}
|
2014-06-20 16:41:51 +02:00
|
|
|
|
2016-05-03 23:45:32 +02:00
|
|
|
private function ytBridgeParseXmlFeed($xml) {
|
2017-07-29 19:28:00 +02:00
|
|
|
foreach($xml->find('entry') as $element) {
|
2017-02-11 16:16:56 +01:00
|
|
|
$title = $this->ytBridgeFixTitle($element->find('title', 0)->plaintext);
|
2016-05-03 23:45:32 +02:00
|
|
|
$author = $element->find('name', 0)->plaintext;
|
|
|
|
$desc = $element->find('media:description', 0)->innertext;
|
2017-04-27 21:40:20 +02:00
|
|
|
|
|
|
|
// Make sure the description is easy on the eye :)
|
|
|
|
$desc = htmlspecialchars($desc);
|
|
|
|
$desc = nl2br($desc);
|
2017-05-02 22:03:38 +02:00
|
|
|
$desc = preg_replace('/(http[s]{0,1}\:\/\/[a-zA-Z0-9.\/\?\&=\-_]{4,})/ims',
|
2017-04-27 21:40:20 +02:00
|
|
|
'<a href="$1" target="_blank">$1</a> ',
|
|
|
|
$desc);
|
|
|
|
|
2016-05-03 23:45:32 +02:00
|
|
|
$vid = str_replace('yt:video:', '', $element->find('id', 0)->plaintext);
|
|
|
|
$time = strtotime($element->find('published', 0)->plaintext);
|
2017-09-24 17:23:00 +02:00
|
|
|
if(strpos($vid, 'googleads') === false)
|
|
|
|
$this->ytBridgeAddItem($vid, $title, $author, $desc, $time);
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
2017-12-29 02:14:11 +01:00
|
|
|
$this->feedName = $this->ytBridgeFixTitle($xml->find('feed > title', 0)->plaintext); // feedName will be used by getName()
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
2014-05-14 12:39:12 +02:00
|
|
|
|
2018-03-19 13:41:53 +01:00
|
|
|
private function ytBridgeParseHtmlListing($html, $element_selector, $title_selector, $add_parsed_items = true) {
|
|
|
|
$limit = $add_parsed_items ? 10 : INF;
|
2017-02-11 16:16:56 +01:00
|
|
|
$count = 0;
|
2018-07-21 14:22:53 +02:00
|
|
|
|
|
|
|
$duration_min = $this->getInput('duration_min') ?: -1;
|
|
|
|
$duration_min = $duration_min * 60;
|
|
|
|
|
|
|
|
$duration_max = $this->getInput('duration_max') ?: INF;
|
|
|
|
$duration_max = $duration_max * 60;
|
|
|
|
|
|
|
|
if($duration_max < $duration_min) {
|
|
|
|
returnClientError('Max duration must be greater than min duration!');
|
|
|
|
}
|
|
|
|
|
2017-07-29 19:28:00 +02:00
|
|
|
foreach($html->find($element_selector) as $element) {
|
|
|
|
if($count < $limit) {
|
2017-02-11 16:16:56 +01:00
|
|
|
$author = '';
|
|
|
|
$desc = '';
|
|
|
|
$time = 0;
|
2016-05-03 23:45:32 +02:00
|
|
|
$vid = str_replace('/watch?v=', '', $element->find('a', 0)->href);
|
2017-08-19 18:51:27 +02:00
|
|
|
$vid = substr($vid, 0, strpos($vid, '&') ?: strlen($vid));
|
2016-05-21 11:45:09 +02:00
|
|
|
$title = $this->ytBridgeFixTitle($element->find($title_selector, 0)->plaintext);
|
2018-07-21 14:22:53 +02:00
|
|
|
|
|
|
|
// The duration comes in one of the formats:
|
|
|
|
// hh:mm:ss / mm:ss / m:ss
|
|
|
|
// 01:03:30 / 15:06 / 1:24
|
|
|
|
$durationText = trim($element->find('span[class="video-time"]', 0)->plaintext);
|
|
|
|
$durationText = preg_replace('/([\d]{1,2})\:([\d]{2})/', '00:$1:$2', $durationText);
|
|
|
|
|
|
|
|
sscanf($durationText, '%d:%d:%d', $hours, $minutes, $seconds);
|
|
|
|
$duration = $hours * 3600 + $minutes * 60 + $seconds;
|
|
|
|
|
|
|
|
if($duration < $duration_min || $duration > $duration_max) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-09-24 17:23:00 +02:00
|
|
|
if($title != '[Private Video]' && strpos($vid, 'googleads') === false) {
|
2018-03-19 13:41:53 +01:00
|
|
|
if ($add_parsed_items) {
|
|
|
|
$this->ytBridgeQueryVideoInfo($vid, $author, $desc, $time);
|
|
|
|
$this->ytBridgeAddItem($vid, $title, $author, $desc, $time);
|
|
|
|
}
|
2016-05-21 11:45:09 +02:00
|
|
|
$count++;
|
|
|
|
}
|
2014-06-20 17:00:36 +02:00
|
|
|
}
|
|
|
|
}
|
2018-03-19 13:41:53 +01:00
|
|
|
return $count;
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
|
|
|
|
2016-05-21 11:45:09 +02:00
|
|
|
private function ytBridgeFixTitle($title) {
|
|
|
|
// convert both Ӓ and " to UTF-8
|
2017-02-11 16:16:56 +01:00
|
|
|
return html_entity_decode($title, ENT_QUOTES, 'UTF-8');
|
2016-05-21 11:45:09 +02:00
|
|
|
}
|
|
|
|
|
2017-04-27 21:40:20 +02:00
|
|
|
private function ytGetSimpleHTMLDOM($url){
|
|
|
|
return getSimpleHTMLDOM($url,
|
2018-03-25 14:01:35 +02:00
|
|
|
$header = array(),
|
|
|
|
$opts = array(),
|
2017-04-27 21:40:20 +02:00
|
|
|
$lowercase = true,
|
|
|
|
$forceTagsClosed = true,
|
|
|
|
$target_charset = DEFAULT_TARGET_CHARSET,
|
|
|
|
$stripRN = false,
|
|
|
|
$defaultBRText = DEFAULT_BR_TEXT,
|
|
|
|
$defaultSpanText = DEFAULT_SPAN_TEXT);
|
|
|
|
}
|
|
|
|
|
2016-08-25 01:24:53 +02:00
|
|
|
public function collectData(){
|
2016-05-03 23:45:32 +02:00
|
|
|
|
|
|
|
$xml = '';
|
|
|
|
$html = '';
|
|
|
|
$url_feed = '';
|
|
|
|
$url_listing = '';
|
2014-05-14 12:39:12 +02:00
|
|
|
|
2017-07-29 19:28:00 +02:00
|
|
|
if($this->getInput('u')) { /* User and Channel modes */
|
2016-08-28 01:25:33 +02:00
|
|
|
$this->request = $this->getInput('u');
|
2017-02-11 16:16:56 +01:00
|
|
|
$url_feed = self::URI . 'feeds/videos.xml?user=' . urlencode($this->request);
|
|
|
|
$url_listing = self::URI . 'user/' . urlencode($this->request) . '/videos';
|
2017-07-29 19:28:00 +02:00
|
|
|
} elseif($this->getInput('c')) {
|
2016-08-28 01:25:33 +02:00
|
|
|
$this->request = $this->getInput('c');
|
2017-02-11 16:16:56 +01:00
|
|
|
$url_feed = self::URI . 'feeds/videos.xml?channel_id=' . urlencode($this->request);
|
|
|
|
$url_listing = self::URI . 'channel/' . urlencode($this->request) . '/videos';
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
2017-02-11 16:16:56 +01:00
|
|
|
|
2017-07-29 19:28:00 +02:00
|
|
|
if(!empty($url_feed) && !empty($url_listing)) {
|
2018-07-21 14:22:53 +02:00
|
|
|
if(!$this->skipFeeds() && $xml = $this->ytGetSimpleHTMLDOM($url_feed)) {
|
2016-05-03 23:45:32 +02:00
|
|
|
$this->ytBridgeParseXmlFeed($xml);
|
2017-07-29 19:28:00 +02:00
|
|
|
} elseif($html = $this->ytGetSimpleHTMLDOM($url_listing)) {
|
2016-05-03 23:45:32 +02:00
|
|
|
$this->ytBridgeParseHtmlListing($html, 'li.channels-content-item', 'h3');
|
2017-02-11 16:16:56 +01:00
|
|
|
} else {
|
|
|
|
returnServerError("Could not request YouTube. Tried:\n - $url_feed\n - $url_listing");
|
|
|
|
}
|
2017-07-29 19:28:00 +02:00
|
|
|
} elseif($this->getInput('p')) { /* playlist mode */
|
2016-08-28 01:25:33 +02:00
|
|
|
$this->request = $this->getInput('p');
|
2018-03-19 13:41:53 +01:00
|
|
|
$url_feed = self::URI . 'feeds/videos.xml?playlist_id=' . urlencode($this->request);
|
2017-02-11 16:16:56 +01:00
|
|
|
$url_listing = self::URI . 'playlist?list=' . urlencode($this->request);
|
2017-04-27 21:40:20 +02:00
|
|
|
$html = $this->ytGetSimpleHTMLDOM($url_listing)
|
2017-02-11 16:16:56 +01:00
|
|
|
or returnServerError("Could not request YouTube. Tried:\n - $url_listing");
|
2018-03-19 13:41:53 +01:00
|
|
|
$item_count = $this->ytBridgeParseHtmlListing($html, 'tr.pl-video', '.pl-video-title a', false);
|
2018-07-21 14:22:53 +02:00
|
|
|
if ($item_count <= 15 && !$this->skipFeeds() && ($xml = $this->ytGetSimpleHTMLDOM($url_feed))) {
|
2018-03-19 13:41:53 +01:00
|
|
|
$this->ytBridgeParseXmlFeed($xml);
|
|
|
|
} else {
|
|
|
|
$this->ytBridgeParseHtmlListing($html, 'tr.pl-video', '.pl-video-title a');
|
|
|
|
}
|
2017-12-29 02:14:11 +01:00
|
|
|
$this->feedName = 'Playlist: ' . str_replace(' - YouTube', '', $html->find('title', 0)->plaintext); // feedName will be used by getName()
|
2018-03-13 12:24:40 +01:00
|
|
|
usort($this->items, function ($item1, $item2) {
|
|
|
|
return $item2['timestamp'] - $item1['timestamp'];
|
|
|
|
});
|
2017-07-29 19:28:00 +02:00
|
|
|
} elseif($this->getInput('s')) { /* search mode */
|
2017-02-11 16:16:56 +01:00
|
|
|
$this->request = $this->getInput('s');
|
|
|
|
$page = 1;
|
|
|
|
if($this->getInput('pa'))
|
2018-06-29 23:55:33 +02:00
|
|
|
$page = (int)preg_replace('/[^0-9]/', '', $this->getInput('pa'));
|
2013-08-15 12:05:58 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
$url_listing = self::URI
|
|
|
|
. 'results?search_query='
|
|
|
|
. urlencode($this->request)
|
|
|
|
. '&page='
|
|
|
|
. $page
|
|
|
|
. '&filters=video&search_sort=video_date_uploaded';
|
2014-06-20 17:00:36 +02:00
|
|
|
|
2017-04-27 21:40:20 +02:00
|
|
|
$html = $this->ytGetSimpleHTMLDOM($url_listing)
|
2017-02-11 16:16:56 +01:00
|
|
|
or returnServerError("Could not request YouTube. Tried:\n - $url_listing");
|
|
|
|
|
2018-04-19 06:03:29 +02:00
|
|
|
$this->ytBridgeParseHtmlListing($html, 'div.yt-lockup', 'h3 > a');
|
2017-12-29 02:14:11 +01:00
|
|
|
$this->feedName = 'Search: ' . str_replace(' - YouTube', '', $html->find('title', 0)->plaintext); // feedName will be used by getName()
|
2017-02-11 16:16:56 +01:00
|
|
|
} else { /* no valid mode */
|
|
|
|
returnClientError("You must either specify either:\n - YouTube
|
|
|
|
username (?u=...)\n - Channel id (?c=...)\n - Playlist id (?p=...)\n - Search (?s=...)");
|
2016-05-03 23:45:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-21 14:22:53 +02:00
|
|
|
private function skipFeeds() {
|
|
|
|
return ($this->getInput('duration_min') || $this->getInput('duration_max'));
|
|
|
|
}
|
|
|
|
|
2014-06-20 17:00:36 +02:00
|
|
|
public function getName(){
|
2017-12-29 02:14:11 +01:00
|
|
|
// Name depends on queriedContext:
|
|
|
|
switch($this->queriedContext) {
|
|
|
|
case 'By username':
|
|
|
|
case 'By channel id':
|
|
|
|
case 'By playlist Id':
|
|
|
|
case 'Search result':
|
2017-12-29 02:19:35 +01:00
|
|
|
return $this->feedName . ' - YouTube'; // We already know it's a bridge, right?
|
|
|
|
default:
|
|
|
|
return parent::getName();
|
2017-12-29 02:14:11 +01:00
|
|
|
}
|
2018-06-23 21:28:27 +02:00
|
|
|
}
|
2013-08-15 12:05:58 +02:00
|
|
|
}
|