- /** YouTube link resolving Originally written by angelsl
- With contributions from Manish Burman http://mburman.com
- With contributions from LouCypher https://github.com/LouCypher
-
- YTGrab is distributed under the GNU LGPL v3 or later and comes with no warranty.
- Full preamble at https://github.com/angelsl/misc-Scripts/blob/master/Greasemonkey/LICENSE.md#ytgrab
-
- //===========DS===========//
- This is a DefSoul MOD for use with hive. All non hive related code is credited to angelsl and contributers above. (My code will have //===========DS===========// above it)
- angelsl's scripts can be found here > https://github.com/angelsl/misc-Scripts
- //===========DS===========\\
-
- // ==UserScript==
- // @name Hive - YouTube to Hive / Local Download
- // @namespace https://openuserjs.org/users/DefSoul/scripts
- // @description Inserts a download button on YouTube video pages and sends to hive -Major fixes
- // @version 1.9 > added ability to send whole playlists to hive
- // @run-at document-end
- // @include http*://www.youtube.com/*
- // @include http*://api.hive.im/api/*
- // @include https://touch.hive.im/account/*
- // @exclude http*://*.google.com/*
- // @exclude http*://*.facebook.com/*
- // @exclude http*://facebook.com/*
- // @exclude about:blank
- // @exclude http*://*.stripe.com/*
- // @require https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js
- // @resource toastrCss http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/css/toastr.min.css
- // @require http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/js/toastr.min.js
- // @grant GM_xmlhttpRequest
- // @grant GM_getValue
- // @grant GM_setValue
- // @grant GM_log
- // @grant GM_getResourceText
- // @grant GM_addStyle
- // @grant unsafeWindow
- // ==/UserScript==
- */
- //===========DS===========//
- var nameB = "YouTube to Hive / Local Download: Test ";
- GM_log(nameB + location.href);
-
- var folderName = "# YouTube #"; // CASE SENSITIVE
- var uploadFolderId;
- var auth;
- auth = GM_getValue("auth");
- var link;
- GM_setValue("ready", "false");
- //GM_deleteValue("auth");
-
- var ru;
- var uploadToHive;
- var uploadPng = "";
- var downloadPng = "";
- function log(str){console.log('%c ' + str, 'background: #000000; color: #FFFFFF');} // CUSTOM LOG
-
- var newCSS = GM_getResourceText ("toastrCss");
- GM_addStyle(newCSS);
-
- toastr.options = {
- "closeButton": false,
- "debug": false,
- "newestOnTop": false,
- "progressBar": false,
- "positionClass": "toast-bottom-right",
- "preventDuplicates": true,
- "onclick": null,
- "showDuration": "300",
- "hideDuration": "1000",
- "timeOut": "12000",
- "extendedTimeOut": "1000",
- "showEasing": "swing",
- "hideEasing": "linear",
- "showMethod": "fadeIn",
- "hideMethod": "fadeOut"
- };
-
-
- $(document).on("click", "#hiveSwitch", function(){
- if ($("#hiveSwitch").attr("src") === uploadPng){
- uploadToHive = false;
- $("#hiveSwitch").attr("src", downloadPng);
- $("#hiveSwitch").attr("title", "Local download activated.");
- document.getElementById('btnDownload').innerHTML = 'Download';
- $("#hiveSwitch").css("right", "9px");
-
- if ($("#watch-action-panels").css("display") == "none")
- document.getElementById("btnDownload").click();
- }
- else{
- uploadToHive = true;
- $("#hiveSwitch").attr("src", uploadPng);
- $("#hiveSwitch").attr("title", "Upload to Hive activated.");
- document.getElementById('btnDownload').innerHTML = ' Upload ';
- $("#hiveSwitch").css("right", "0px");
-
- if ($("#watch-action-panels").css("display") == "none")
- document.getElementById("btnDownload").click();
- }
-
- });
- //===========DS===========\\
-
- if (typeof unsafeWindow === 'undefined' || typeof unsafeWindow.ytplayer === 'undefined') {
- var p = document.createElement('p');
- p.setAttribute('onclick', 'return window;');
- unsafeWindow = p.onclick();
- }
-
- function main(decipher) {
- var dashmpd = unsafeWindow.ytplayer.config.args.dashmpd, mpbsrgx = /\/s\/([\w\.]+)/, mpbs;
- if (typeof dashmpd !== 'undefined') {
- mpbs = mpbsrgx.exec(dashmpd); if(mpbs) dashmpd = dashmpd.replace(mpbsrgx, "/signature/"+decipher(mpbs[1]));
- GM_xmlhttpRequest({method: "GET", url: dashmpd, onload: function (t) { main2(t.responseText, decipher); }});
- } else main2(false, decipher);
- }
-
- function main2(dashmpd, decipher) {
- "use strict";
- var
- uriencToMap = function (s) {
- var n = {}, a = s.split("&"), idy, c;
- for (idy = 0; idy < a.length; idy++) {
- c = a[idy].split("=");
- n[c[0]] = decodeURIComponent(c[1]);
- }
- return n;
- },
- uwyca = unsafeWindow.ytplayer.config.args,
- title = uwyca.title.replace(/[\/\\\:\*\?\"<\>\|]/g, ""),
- fmtrgx = /^[\-\w+]+\/(?:x-)?([\-\w+]+)/,
- fmt_map = {}, idx, idz, n, a, qual, fmt, fmt_list, map, uefmss, dashlist, ul, q, div,
- type, itag, maporder, fpsa, fpsb, fpsw = false;
-
- fmt_list = uwyca.fmt_list.split(",");
- for (idx = 0; idx < fmt_list.length; idx++) {
- a = fmt_list[idx].split("/");
- fmt_map[a[0]] = a[1].split("x")[1] + "p";
- }
-
- map = {};
- uefmss = uwyca.url_encoded_fmt_stream_map.split(",");
- for (idx = 0; idx < uefmss.length; idx++) {
- n = uriencToMap(uefmss[idx]);
- qual = fmt_map[n.itag];
-
- if (!(qual in map)) { map[qual] = []; }
- fmt = fmtrgx.exec(n.type);
- map[qual].push($("<a>" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Quality: " + n.quality + " | Mime: " + n.type));
- }
-
- dashlist = uwyca.adaptive_fmts;
- if (typeof dashlist !== 'undefined') {
- dashlist = dashlist.split(",");
- for (idx = 0; idx < dashlist.length; idx++) {
- n = uriencToMap(dashlist[idx]);
- qual = n.type.indexOf("audio/") === 0 ? "Audio" : (("size" in n) ? (n.size.split('x')[1] + 'p' + n.fps) : (n.itag in fmt_map) ? (fmt_map[n.itag]) : ("Unknown"));
-
- if (!(qual in map)) { map[qual] = []; }
- fmt = fmtrgx.exec(n.type);
- if (parseInt(n.fps) == 1) fpsw = 1;
- map[qual].push($("<a>DASH" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Bitrate: " + n.bitrate + " | Mime: " + n.type + " | Res: " + n.size + " | FPS: " + n.fps));
- }
- }
-
- if (dashmpd !== false) {
- dashmpd = $($.parseXML(dashmpd));
- dashmpd.find("AdaptationSet").each(function() {
- q = $(this); type = q.attr("mimeType");
- q.children("Representation").each(function() {
- n = $(this); itag = n.attr("id");
- qual = type.indexOf("audio/") === 0 ? "Audio" : (n.attr("height") + 'p' + n.attr("frameRate"));
- if (!(qual in map)) { map[qual] = []; }
- fmt = fmtrgx.exec(type);
- if (parseInt(n.attr("frameRate")) == 1) fpsw = 1;
- map[qual].push($("<a>MPD" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.children("BaseURL").text() + "&title=" + title).attr("title", "Format ID: " + itag + " | Bitrate: " + n.attr("bandwidth") + " | Mime: " + type + (type.indexOf("audio/") === 0 ? " | Sample Rate: " + n.attr("audioSamplingRate") : " | Res: " + n.attr("width") + 'x' + n.attr("height") + " | FPS: " + n.attr("frameRate"))));
- });
- });
- }
-
- maporder = Object.keys(map);
- maporder.sort(function(a,b) {
- if((a == "Audio" && b == "Unknown") || (b == "Audio" && a != "Unknown")) return -1;
- if ((b == "Audio" && a == "Unknown") || (a == "Audio" && b != "Unknown")) return 1;
- fpsa = a.split('p')[1] || 0; fpsb = b.split('p')[1] || 0; if (fpsa != fpsb) return parseInt(fpsb)-parseInt(fpsa);
- return parseInt(b)-parseInt(a); });
- ul = $("<ul class=\"watch-extras-section\" />");
- for (n = 0; n < maporder.length; ++n) {
- q = maporder[n];
- if (map[q].length < 1) { continue; }
- div = $("<div class=\"content\" />").append(map[q][0]);
- for (idz = 1; idz < map[q].length; idz++) {
- div.append(" ").append(map[q][idz]);
- }
- ul.append($("<li><h4 class=\"title\" style=\"font-weight: bold; color: #333333;\">" + q + "</h4></li>").append(div));
- }
-
- $("#action-panel-share").after($("<div id=\"action-panel-sldownload\" class=\"action-panel-content hid\" data-panel-loaded=\"true\" />").append(ul));
- $("#watch8-secondary-actions").find("> div").eq(1).after($('<button class="yt-uix-button yt-uix-button-size-default yt-uix-button-opacity action-panel-trigger yt-uix-button-opacity yt-uix-tooltip" style="text-align: center;" type="button" onclick=";return false;" title="" id="btnDownload" data-trigger-for="action-panel-sldownload" data-button-toggle="true"><span class="yt-uix-button-content">Upload</span></button>')).size();
- //===========DS===========//
- //$("#hiveSwitch").css("display", "block");
- $("#watch8-secondary-actions").find("> div").eq(1).after($('<img title="Upload to Hive activated." src="' + uploadPng + '" type="button" onclick=";return false;" title="" id="hiveSwitch" style="right: 0px; bottom: 41px; z-index: 9999999; cursor: pointer; position: absolute; display: block; height: 50px; width: 50px; padding-left: 5px;" data-button-toggle="true"><span class=""></span></button>')).size();
-
- //===========DS===========\\
- if (fpsw) ul.after($("<p style='color: green;'>At this time Hive only accepts Mp4 & Flv video files, the other formats are for local downloading.</p>"));
- }
-
- function run() {
- if (typeof unsafeWindow.ytplayer !== 'undefined')
- { GM_xmlhttpRequest({method: "GET", url: unsafeWindow.ytplayer.config.assets.js.replace(/^\/\//, "https://"), onload: function (t) { main((function (u) {
- "use strict"; var sres = /function ([a-zA-Z$0-9]+)\(a\)\{a=a\.split\(""\);([a-zA-Z0-9]*)\.?.*?return a\.join\(""\)\};/g.exec(u);
- if (!sres) { return function (v) { return v; }; }
- return eval("(function(s){" + (sres[2] !== "" ? (new RegExp("var " + sres[2] + "={.+?}};", "g").exec(u)[0]) : "") + sres[0] + "return " + sres[1] + "(s);})");
- }(t.responseText))); }}); }
- }
-
- //DS//
- function run2(val) {
- log("run2 running");
- GM_xmlhttpRequest({
- method: "GET",
- url: val,
- onload: function(t){
-
- json = JSON.parse(t.responseText);
-
- //for(var key in json) {
- // var value = json[key];
- // log(value);
- //}
- }});
- }
-
- setInterval(function(){
- if ($(".playlist-actions").length && !$("#PlaylistToHive").length){
- $(".playlist-actions").append('<button id="PlaylistToHive" class="yt-uix-button yt-uix-button-size-default yt-uix-button-default yt-uix-button-has-icon no-icon-markup yt-uix-playlistlike yt-uix-tooltip" type="button" onclick=";return false;" aria-label="To Hive" title="To Hive" data-like-tooltip="Save to Playlists" data-unlike-tooltip="Remove" data-like-label="Save" data-unlike-label="Saved" data-tooltip-text="To Hive" aria-labelledby="yt-uix-tooltip95-arialabel" data-tooltip-hide-timer="235"><span class="yt-uix-button-content">To Hive</span></button>');
- }
- }, 1000);
-
- //DS\\
-
- waitForKeyElements("#watch8-secondary-actions", run);
-
- //===========DS===========/
-
- function createFolder(uploadFolderName){
- GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
- "method": "get",
- "url": "https://api.hive.im/api/hive/get/",
- "headers": {
- 'Content-Type': 'application/x-www-form-urlencoded;',
- 'Authorization': auth,
- 'Client-Type': 'Browser',
- 'Client-Version': '0.1',
- 'Referer': 'https://touch.hive.im/myfiles/videos',
- 'Origin': 'https://touch.hive.im/'
- },
- "onload": function(data){
- var r = data.responseText;
- var json = JSON.parse(r);
-
- for (var i = 0; i < json.data.length; i++){
- var id;
-
- if (json.data[i].title === "Videos"){ // FINDS INITIAL VIDEOS FOLDER ID
- //log("we got a video ova here", "green");
-
- parentId = json.data[i].parentId;
- id = json.data[i].id;
-
- GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
- "method": "post",
- "url": "https://api.hive.im/api/hive/get-children/",
- "data": "&parentId=" + id + "&limit=1000",
- "headers": {
- 'Content-Type': 'application/x-www-form-urlencoded;',
- 'Authorization': auth,
- 'Client-Type': 'Browser',
- 'Client-Version': '0.1',
- 'Referer': 'https://touch.hive.im/',
- 'Origin': 'https://touch.hive.im/'
- },
- "onload": function(data){
- var r = data.responseText;
- var json = JSON.parse(r);
- var hasFolderIndex;
-
- Object.keys(json.data).forEach(function(key) {
- //log(json.data[key].title, "blue");
- hasFolderIndex += json.data[key].title;
-
- if (json.data[key].title === uploadFolderName){
- uploadFolderId = json.data[key].id;
- log("<" + uploadFolderName + "> Already exists. " + uploadFolderId, "green");
- //return json.data[key].id;
- }
- });
-
- if (hasFolderIndex.indexOf(uploadFolderName) == -1){ // SEARCHES VIDEOS FOLDER TO SEE IF uploadFolderName EXISTS
- log("does not contain: " + uploadFolderName, "red");
-
- GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
- "method": "post",
- "url": "https://api.hive.im/api/hive/create/",
- "data": "filename=" + uploadFolderName + "&parent=" + id + "&locked=false",
- "headers": {
- 'Content-Type': 'application/x-www-form-urlencoded;',
- 'Authorization': auth,
- 'Client-Type': 'Browser',
- 'Client-Version': '0.1',
- 'Referer': 'https://touch.hive.im/',
- 'Origin': 'https://touch.hive.im/'
- },
- "onload": function(data){
- var r = data.responseText;
- var json = JSON.parse(r);
-
- uploadFolderId = json.data.id;
-
- log("Create folder <" + uploadFolderName + "> " + json.data.id);
- return json.data.id;
- }
- });
- }
- else{
- //log("does contain: " + uploadFolderName, "green");
- }
- }
- });
- //log(parentId + "\n" + currentId);
- }
-
- //log(item, "blue");
- }
-
- //log(r, "blue");
- }
- });
- }
-
- function cdReq(href, nameT, folderId){
- log("cdReq start: " + href);
- GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
- "method": "post",
- "url": "https://api.hive.im/api/transfer/add/",
- "data": "remoteUrl=" + window.btoa(href) + "&parentId=" + folderId,
- //"data": "remoteUrl=" + window.btoa(href),
- "headers": {
- 'Content-Type': 'application/x-www-form-urlencoded;',
- 'Authorization': GM_getValue("auth"),
- 'Client-Type': 'Browser',
- 'Client-Version': '0.1',
- 'Referer': 'https://touch.hive.im/',
- 'Origin': 'https://touch.hive.im/'
- },
- "onload": function(data){
- var r = data.responseText;
- var json = JSON.parse(r);
-
- if (json.status === "success"){
- toastr.success(nameT, "Status: " + json.data.status);
-
- log("========= " + nameT + " success =========", "green");
- log("Job ID: " + json.data.jobId, "blue");
- log("Data Status: " + json.data.status, "blue");
- log("Folder Id: " + folderId, "blue");
- log("", "red");
- }
- else{
- if (json.message === "quotaExceeded"){
- toastr.warning(nameT, "Quota Exceeded");
- }
- else if (json.message === "securityViolation"){
- toastr.error(nameT, "Security Violation");
- }
-
- log("========= " + nameT + " error =========", "green");
- log("Message: " + json.message, "blue");
- log("", "red");
- }
-
- //log("cdReq >" + data.responseText);
-
- //transferItemsList(); // GO GET ITEMS IN CURRENT TRANSFER LIST
- }
- });
- }
-
- $(document).on("click", "#PlaylistToHive", function(e){ // MAIN CLICK EVENT
- e.preventDefault();
-
- toastr.warning("Extracting links.", "Please don't navigate from page!");
-
- var tiles = document.getElementsByClassName("yt-uix-tile");
- var titlesClass = document.getElementsByClassName("pl-video-title-link");
- var titles = [];
- var vids = []; // CONTAINS ALL COMPLETE URLS OF ALL ITEMS IN PLAYLIST
- var mp4s = [];
-
- for (var i = 0; i < tiles.length; i++){
- var r = $(titlesClass[i]).html();
- r = r.replace(/(\r\n|\n|\r)/gm,"");
- r = r.trim();
- r = r.replace(/[`~!@#$%^&*()_|+\=÷¿?;:'",.<>\{\}\[\]\\\/]/gi, '%20');
- r = r.replace(/ /g, "%20");
- //r = "&title=" + r;
-
- //log(r);
-
- titles.push(r); // CREATES ARRAY OF VIDEO TITLES
-
- vids.push("https://www.youtube.com/watch?v=" + $(tiles[i]).attr("data-video-id")); // CREATES ARRAY OF VIDEO URLS
- }
-
- var jjj = 0;
- for (var jI = 0; jI < vids.length; jI++){
- var toastTitle;
-
- extract(vids[jI]).done(function (result) {
-
- for (var j = 0; j < result.formats.length; j++){
- if (result.formats[j].ext === "mp4" && typeof result.formats[j].format_note == "undefined"){
- toastTitle = titles[jjj];
- toastTitle = toastTitle.replace(/%20/g, " ");
-
- mp4s = [];
- mp4s.push(result.formats[j].url + "&title=" + titles[jjj]);
- }
- }
-
- //log("MP4S 1: >>" + mp4s[0], "blue"); // HIGHEST QUALITY MP4
- cdReq(mp4s[0], toastTitle, uploadFolderId);
-
- jjj++;
-
- setTimeout(function(){
- if (jjj === jI){
- toastr.info("Finished!");
- }
- }, 5000);
- });
- }
-
- });
-
- if (window.top === window.self) {
- //=========MAIN WINDOW=========//
- if (document.location.href.indexOf("touch.hive.im") !== -1){
- return;
- }
-
- createFolder(folderName);
-
- if (!$("#iframeHive").length || typeof auth == "undefined"){
- var iframe = document.createElement('iframe');
- iframe.id = "iframeHive";
- iframe.src = "https://touch.hive.im/account/?1";
- iframe.style = "height: 0px; width: 0px; display: none; overflow:hidden";
- document.body.appendChild(iframe);
- $("#iframeHive").attr("style", "height: 0px; width: 0px; display: none; overflow:hidden");
- //$("#iframeHive").attr("style", "height: 600px; width: 600px; display: block; overflow:hidden");
- log("iframe created! " + nameB);
- }
-
- var onceB = 0;
- setInterval(function(){
- //log("AA: " + auth);
- if (onceB === 0 && typeof auth !== "undefined"){
- GM_setValue("ready", "true")
- GM_setValue("auth", auth);
- $("#iframeHive").remove();
- //log("TRUE: " + auth);
- }
-
- if (onceB === 0 && GM_getValue("ready") == "true"){
- onceB = 1;
-
- auth = GM_getValue("auth");
- log("A: " + auth);
-
- $("#iframeHive").remove();
- //init();
- }
- }, 250);
-
- $(document).on("click", "a", function(evt){ // MAIN CLICK EVENT
- if ($(this).attr('href').indexOf('googlevideo') !== -1){
- if (uploadToHive === false)
- return;
-
- log($("#hiveSwitch").attr("src"));
-
- evt.preventDefault();
- ru = $(this).attr('href');
- //log("pre: " + ru);
- ru = ru.replace(/ /g, "%20");
-
- //log("post: " + ru);
- var vidTitle = $("#eow-title").attr("title");
- cdReq(ru, vidTitle, uploadFolderId);
- }
- });
- }
- else
- {
- //=========IFRAME WINDOW=========//
- try{
- auth = unsafeWindow.account.token;
- }
- catch(err){}
-
- var once = 0;
- setInterval(function(){ // EVENT FOR WHEN PAGE IS LOADED // RUNS ONCE
- if (once === 0 && $("#username").text().indexOf("My Account") !== -1){
- once = 1;
- log("ready");
-
- auth = unsafeWindow.account.token;
- GM_setValue("auth", unsafeWindow.account.token);
- GM_setValue("ready", "true");
-
- }
- else if (once === 1 && auth == "undefined"){
- GM_setValue("ready", "false");
- try{
- auth = unsafeWindow.account.token;
- }
- catch(err){}
- }
- }, 200);
- }
- //===========DS===========\\
-
- // START YT-LINKS CODE //
- var YT_FORMATS = {
- '5': {'ext': 'flv', 'width': 400, 'height': 240},
- '6': {'ext': 'flv', 'width': 450, 'height': 270},
- '13': {'ext': '3gp'},
- '17': {'ext': '3gp', 'width': 176, 'height': 144},
- '18': {'ext': 'mp4', 'width': 640, 'height': 360},
- '22': {'ext': 'mp4', 'width': 1280, 'height': 720},
- '34': {'ext': 'flv', 'width': 640, 'height': 360},
- '35': {'ext': 'flv', 'width': 854, 'height': 480},
- '36': {'ext': '3gp', 'width': 320, 'height': 240},
- '37': {'ext': 'mp4', 'width': 1920, 'height': 1080},
- '38': {'ext': 'mp4', 'width': 4096, 'height': 3072},
- '43': {'ext': 'webm', 'width': 640, 'height': 360},
- '44': {'ext': 'webm', 'width': 854, 'height': 480},
- '45': {'ext': 'webm', 'width': 1280, 'height': 720},
- '46': {'ext': 'webm', 'width': 1920, 'height': 1080},
- '59': {'ext': 'mp4', 'width': 854, 'height': 480},
- '78': {'ext': 'mp4', 'width': 854, 'height': 480},
-
- // 3d videos
- '82': {'ext': 'mp4', 'height': 360, 'format_note': '3D', 'preference': -20},
- '83': {'ext': 'mp4', 'height': 480, 'format_note': '3D', 'preference': -20},
- '84': {'ext': 'mp4', 'height': 720, 'format_note': '3D', 'preference': -20},
- '85': {'ext': 'mp4', 'height': 1080, 'format_note': '3D', 'preference': -20},
- '100': {'ext': 'webm', 'height': 360, 'format_note': '3D', 'preference': -20},
- '101': {'ext': 'webm', 'height': 480, 'format_note': '3D', 'preference': -20},
- '102': {'ext': 'webm', 'height': 720, 'format_note': '3D', 'preference': -20},
-
- // Apple HTTP Live Streaming
- '92': {'ext': 'mp4', 'height': 240, 'format_note': 'HLS', 'preference': -10},
- '93': {'ext': 'mp4', 'height': 360, 'format_note': 'HLS', 'preference': -10},
- '94': {'ext': 'mp4', 'height': 480, 'format_note': 'HLS', 'preference': -10},
- '95': {'ext': 'mp4', 'height': 720, 'format_note': 'HLS', 'preference': -10},
- '96': {'ext': 'mp4', 'height': 1080, 'format_note': 'HLS', 'preference': -10},
- '132': {'ext': 'mp4', 'height': 240, 'format_note': 'HLS', 'preference': -10},
- '151': {'ext': 'mp4', 'height': 72, 'format_note': 'HLS', 'preference': -10},
-
- // DASH mp4 video
- '133': {'ext': 'mp4', 'height': 240, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '134': {'ext': 'mp4', 'height': 360, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '135': {'ext': 'mp4', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '136': {'ext': 'mp4', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '137': {'ext': 'mp4', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '138': {'ext': 'mp4', 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, // Height can vary (https://github.com/rg3/youtube-dl/issues/4559)
- '160': {'ext': 'mp4', 'height': 144, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '264': {'ext': 'mp4', 'height': 1440, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '298': {
- 'ext': 'mp4',
- 'height': 720,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'h264'
- },
- '299': {
- 'ext': 'mp4',
- 'height': 1080,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'h264'
- },
- '266': {
- 'ext': 'mp4',
- 'height': 2160,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'vcodec': 'h264'
- },
-
- // Dash mp4 audio
- '139': {
- 'ext': 'm4a',
- 'format_note': 'DASH audio',
- 'acodec': 'aac',
- 'vcodec': 'none',
- 'abr': 48,
- 'preference': -50,
- 'container': 'm4a_dash'
- },
- '140': {
- 'ext': 'm4a',
- 'format_note': 'DASH audio',
- 'acodec': 'aac',
- 'vcodec': 'none',
- 'abr': 128,
- 'preference': -50,
- 'container': 'm4a_dash'
- },
- '141': {
- 'ext': 'm4a',
- 'format_note': 'DASH audio',
- 'acodec': 'aac',
- 'vcodec': 'none',
- 'abr': 256,
- 'preference': -50,
- 'container': 'm4a_dash'
- },
-
- // Dash webm
- '167': {
- 'ext': 'webm',
- 'height': 360,
- 'width': 640,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '168': {
- 'ext': 'webm',
- 'height': 480,
- 'width': 854,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '169': {
- 'ext': 'webm',
- 'height': 720,
- 'width': 1280,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '170': {
- 'ext': 'webm',
- 'height': 1080,
- 'width': 1920,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '218': {
- 'ext': 'webm',
- 'height': 480,
- 'width': 854,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '219': {
- 'ext': 'webm',
- 'height': 480,
- 'width': 854,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'container': 'webm',
- 'vcodec': 'VP8',
- 'preference': -40
- },
- '278': {
- 'ext': 'webm',
- 'height': 144,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'container': 'webm',
- 'vcodec': 'VP9'
- },
- '242': {'ext': 'webm', 'height': 240, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '243': {'ext': 'webm', 'height': 360, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '244': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '245': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '246': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '247': {'ext': 'webm', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '248': {'ext': 'webm', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '271': {'ext': 'webm', 'height': 1440, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '272': {'ext': 'webm', 'height': 2160, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40},
- '302': {
- 'ext': 'webm',
- 'height': 720,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'VP9'
- },
- '303': {
- 'ext': 'webm',
- 'height': 1080,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'VP9'
- },
- '308': {
- 'ext': 'webm',
- 'height': 1440,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'VP9'
- },
- '313': {
- 'ext': 'webm',
- 'height': 2160,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'vcodec': 'VP9'
- },
- '315': {
- 'ext': 'webm',
- 'height': 2160,
- 'format_note': 'DASH video',
- 'acodec': 'none',
- 'preference': -40,
- 'fps': 60,
- 'vcodec': 'VP9'
- },
-
- // Dash webm audio
- '171': {'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'abr': 128, 'preference': -50},
- '172': {'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'abr': 256, 'preference': -50},
-
- // Dash webm audio with opus inside
- '249': {
- 'ext': 'webm',
- 'vcodec': 'none',
- 'format_note': 'DASH audio',
- 'acodec': 'opus',
- 'abr': 50,
- 'preference': -50
- },
- '250': {
- 'ext': 'webm',
- 'vcodec': 'none',
- 'format_note': 'DASH audio',
- 'acodec': 'opus',
- 'abr': 70,
- 'preference': -50
- },
- '251': {
- 'ext': 'webm',
- 'vcodec': 'none',
- 'format_note': 'DASH audio',
- 'acodec': 'opus',
- 'abr': 160,
- 'preference': -50
- },
-
- // RTMP (unnamed)
- '_rtmp': {'protocol': 'rtmp'},
- }
-
- // QueryString - begin
-
- // This is public domain code written in 2011 by Jan Wolter and distributed
- // for free at http://unixpapa.com/js/querystring.html
- //
- // Query String Parser
- //
- // qs= new QueryString()
- // qs= new QueryString(string)
- //
- // Create a query string object based on the given query string. If
- // no string is given, we use the one from the current page by default.
- //
- // qs.value(key)
- //
- // Return a value for the named key. If the key was not defined,
- // it will return undefined. If the key was multiply defined it will
- // return the last value set. If it was defined without a value, it
- // will return an empty string.
- //
- // qs.values(key)
- //
- // Return an array of values for the named key. If the key was not
- // defined, an empty array will be returned. If the key was multiply
- // defined, the values will be given in the order they appeared on
- // in the query string.
- //
- // qs.keys()
- //
- // Return an array of unique keys in the query string. The order will
- // not necessarily be the same as in the original query, and repeated
- // keys will only be listed once.
- //
- // QueryString.decode(string)
- //
- // This static method is an error tolerant version of the builtin
- // function decodeURIComponent(), modified to also change pluses into
- // spaces, so that it is suitable for query string decoding. You
- // shouldn't usually need to call this yourself as the value(),
- // values(), and keys() methods already decode everything they return.
- //
- // Note: W3C recommends that ; be accepted as an alternative to & for
- // separating query string fields. To support that, simply insert a semicolon
- // immediately after each ampersand in the regular expression in the first
- // function below.
-
- function QueryString(qs) {
- this.dict = {};
-
- // If no query string was passed in use the one from the current page
- if (!qs) qs = location.search;
-
- // Delete leading question mark, if there is one
- if (qs.charAt(0) == '?') qs = qs.substring(1);
-
- // Parse it
- var re = /([^=&]+)(=([^&]*))?/g;
- while (match = re.exec(qs)) {
- var key = decodeURIComponent(match[1].replace(/\+/g, ' '));
- var value = match[3] ? QueryString.decode(match[3]) : '';
- if (this.dict[key])
- this.dict[key].push(value);
- else
- this.dict[key] = [value];
- }
- }
-
- QueryString.decode = function (s) {
- s = s.replace(/\+/g, ' ');
- s = s.replace(/%([EF][0-9A-F])%([89AB][0-9A-F])%([89AB][0-9A-F])/gi,
- function (code, hex1, hex2, hex3) {
- var n1 = parseInt(hex1, 16) - 0xE0;
- var n2 = parseInt(hex2, 16) - 0x80;
- if (n1 == 0 && n2 < 32) return code;
- var n3 = parseInt(hex3, 16) - 0x80;
- var n = (n1 << 12) + (n2 << 6) + n3;
- if (n > 0xFFFF) return code;
- return String.fromCharCode(n);
- });
- s = s.replace(/%([CD][0-9A-F])%([89AB][0-9A-F])/gi,
- function (code, hex1, hex2) {
- var n1 = parseInt(hex1, 16) - 0xC0;
- if (n1 < 2) return code;
- var n2 = parseInt(hex2, 16) - 0x80;
- return String.fromCharCode((n1 << 6) + n2);
- });
- s = s.replace(/%([0-7][0-9A-F])/gi,
- function (code, hex) {
- return String.fromCharCode(parseInt(hex, 16));
- });
- return s;
- };
-
- QueryString.prototype.value = function (key) {
- var a = this.dict[key];
- return a ? a[a.length - 1] : undefined;
- };
-
- QueryString.prototype.values = function (key) {
- var a = this.dict[key];
- return a ? a : [];
- };
-
- QueryString.prototype.keys = function () {
- var a = [];
- for (var key in this.dict)
- a.push(key);
- return a;
- };
-
- // QueryString - end
-
- var Queue = function () {
- var previous = new $.Deferred().resolve();
-
- return function (fn, fail) {
- return previous = previous.then(fn, fail || fn);
- };
- };
-
- var queue = Queue(); // lower case for idiomatic use
-
- var LAST_PLAYER_URL = null;
- var LAST_FUNC = null;
-
- function log(s) {
- try {
- console.log(s);
- } catch (ignore) {
- }
- }
-
- function download(url) {
- log('Downloading webpage ' + url);
-
- var deferred = $.Deferred();
-
- //var userAgent = navigator.userAgent;
- var userAgent = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36';
-
- $.get('http://query.yahooapis.com/v1/public/yql', {
- q: 'select * from xClient where url="' + url + '" and ua="' + userAgent + '"',
- format: 'json',
- env: 'store://datatables.org/alltableswithkeys',
- callback: ''
- }).done(function (data) {
- try {
- deferred.resolve(data.query.results.resources.content);
- } catch (e) {
- log(e);
- deferred.resolve(null);
- }
- });
-
- return deferred.promise();
- }
-
- function extractId(url) {
- var r = /^http(s?):\/\/www\.youtube\.com\/watch\?v=(.+)/.exec(url);
- return r !== null ? r[2] : null;
- }
-
- function searchRegex(regex, string, defaultValue) {
- var r = regex.exec(string);
-
- if (r !== null) {
- return r[1];
- } else {
- return defaultValue;
- }
- }
-
- function parseQS(s) {
- var qs = new QueryString(s);
-
- var obj = {};
-
- var keys = qs.keys();
- var size = keys.length;
-
- for (var i = 0; i < size; i++) {
- var k = keys[i];
- obj[k] = qs.values(k);
- }
-
- return obj;
- }
-
- function decryptSignature(s, playerUrl) {
- var deferred = $.Deferred();
-
- if (playerUrl === null) {
- log('Cannot decrypt signature without player_url');
- deferred.resolve(null);
- }
-
- if (playerUrl.indexOf('//') === 0) {
- playerUrl = 'https:' + playerUrl;
- }
-
- if (LAST_PLAYER_URL === playerUrl && LAST_FUNC !== null) {
- var func = LAST_FUNC;
- var signature = func(s);
- deferred.resolve(signature);
- } else {
- download(playerUrl).done(function (jscode) {
- var func = null;
-
- if (LAST_PLAYER_URL === playerUrl && LAST_FUNC !== null) {
- func = LAST_FUNC;
- } else {
-
- var r = /\.sig\|\|([a-zA-Z0-9$]+)\(/.exec(jscode);
- if (r === null) {
- log("Couldn't find the signature code with regex");
- }
-
- var funcname = r[1];
-
- function shortcut(jscode) {
- var p = jscode.split('function ' + funcname + '(');
- if (p.length !== 2) {
- return null;
- }
-
- var i1 = p[0].lastIndexOf('};var ');
- if (i1 === -1) {
- return null;
- }
-
- var p1 = p[0].substr(i1 + 2);
-
- var i2 = p[1].indexOf('};');
- if (i2 === -1) {
- return null;
- }
-
- var p2 = p[1].substr(0, i2 + 2);
-
- return p1 + 'function ' + funcname + '(' + p2;
- }
-
- var temp = shortcut(jscode);
- if (temp !== null) {
- jscode = temp;
- }
-
- var ast = esprima.parse(jscode);
-
- function traverse(object, level, visitor) {
- var key, child;
-
- if (visitor.call(null, object) === false) {
- return;
- }
-
- if (level > 8) {
- return;
- }
-
- for (key in object) {
- if (object.hasOwnProperty(key)) {
- child = object[key];
- if (typeof child === 'object' && child !== null) {
- traverse(child, level + 1, visitor);
- }
- }
- }
- }
-
- traverse(ast, 0, function (node) {
- if (node.type === 'FunctionDeclaration' && node.id.name == funcname) {
- func = eval('(' + escodegen.generate(node) + ')');
- }
-
- if (node.type === 'VariableDeclarator') {
- try {
- eval(escodegen.generate(node));
- } catch (ignore) {
- }
- }
- });
-
- LAST_PLAYER_URL = playerUrl;
- LAST_FUNC = func;
- }
-
- var signature = func(s);
- deferred.resolve(signature);
- });
- }
-
- return deferred.promise();
- }
-
- function parseDashManifest(video_id, dash_manifest_url, player_url, age_gate) {
- var deferred = $.Deferred();
-
- var r = /\/s\/([a-fA-F0-9\.]+)/.exec(dash_manifest_url);
-
- if (r !== null) {
- var s = r[1];
-
- var formats = [];
-
- decryptSignature(s, player_url).done(function (dec_s) {
- dash_manifest_url = dash_manifest_url.replace(new RegExp('/s/' + s), '/signature/' + dec_s);
-
- download(dash_manifest_url).done(function (dash_doc) {
- dash_doc = $.parseXML(dash_doc);
-
- $(dash_doc).find('AdaptationSet').each(function (index, elemSet) {
- var mimeType = $(elemSet).attr('mimeType');
-
- $(elemSet).find('Representation').each(function (index, elemRep) {
- var url_el = $(elemRep).find('BaseURL');
-
- if (mimeType.indexOf('audio/') === 0 || mimeType.indexOf('video/') === 0) {
-
- var format_id = $(elemRep).attr('id');
- var video_url = $(url_el).text();
- var filesize = parseInt($(url_el).attr('yt:contentLength'));
-
- var f = {
- format_id: format_id,
- url: video_url,
- widt: parseInt($(elemRep).attr('width')),
- height: parseInt($(elemRep).attr('height')),
- filesize: filesize
- }
-
- formats.push(f);
- }
- });
- });
-
- deferred.resolve({
- dash_manifest_url: dash_manifest_url,
- formats: formats
- });
- });
- });
- } else {
- deferred.resolve(null);
- }
-
- return deferred.promise();
- }
-
- function getSubtitles(videoId) {
- var deferred = $.Deferred();
-
- download('https://video.google.com/timedtext?hl=en&type=list&v=' + videoId).done(function (subsDoc) {
- subsDoc = $.parseXML(subsDoc);
-
- var subLangList = {};
-
- $(subsDoc).find('track').each(function (index, track) {
- var lang = $(track).attr('lang_code');
- if (subLangList.hasOwnProperty(lang)) {
- return;
- }
-
- var subFormats = [];
-
- ['sbv', 'vtt', 'srt'].forEach(function (ext) {
- var params = $.param({
- lang: lang,
- v: videoId,
- fmt: ext,
- name: $(track).attr('name'),
- });
-
- subFormats.push({
- 'url': 'https://www.youtube.com/api/timedtext?' + params,
- 'ext': ext,
- });
- });
-
- subLangList[lang] = subFormats;
- });
-
- deferred.resolve(subLangList);
- }).fail(function () {
- deferred.resolve(null);
- });
-
- return deferred.promise();
- }
-
- function extractSupport(video_id, video_webpage, age_gate, embed_webpage, video_info) {
- var deferred = $.Deferred();
-
- function fail(s) {
- log(s);
- deferred.resolve(null);
- return deferred.promise();
- }
-
- if (!video_info.hasOwnProperty('token')) {
- if (video_info.hasOwnProperty('reason')) {
- return fail('YouTube said: ' + video_info['reason'][0]);
- }
- else {
- return fail('"token" parameter not in video info for unknown reason');
- }
- }
-
- var view_count = 0;
- if (video_info.hasOwnProperty('view_count')) {
- view_count = parseInt(video_info['view_count'][0]);
- }
-
- // Check for "rental" videos
- if (video_info.hasOwnProperty('ypc_video_rental_bar_text') && !video_info.hasOwnProperty('author')) {
- return fail('"rental" videos not supported');
- }
-
- //Start extracting information
- //self.report_information_extraction(video_id)
-
- // uploader
- if (!video_info.hasOwnProperty('author')) {
- return fail('Unable to extract uploader name');
- }
-
- var video_uploader = decodeURIComponent(video_info['author'][0]);
-
- // uploader_id
- var video_uploader_id = null;
-
- var mobj = /<link itemprop="url" href="http:\/\/www.youtube.com\/(?:user|channel)\/([^"]+)">/.exec(video_webpage);
- if (mobj !== null) {
- video_uploader_id = mobj[1];
- }
- else {
- //return fail('unable to extract uploader nickname');
- log('unable to extract uploader nickname');
- }
-
- // title
- var video_title = '_';
- if (video_info.hasOwnProperty('title')) {
- video_title = video_info['title'][0];
- }
- else {
- return fail('Unable to extract video title');
- }
-
- // upload date
- var upload_date = null;
- mobj = /id="eow-date.*?>(.*?)<\/span>/.exec(video_webpage);
- if (mobj === null) {
- mobj = /id="watch-uploader-info".*?>.*?(?:Published|Uploaded|Streamed live) on (.*?)<\/strong>/.exec(video_webpage);
- }
- if (mobj !== null) {
- upload_date = new Date(mobj[1]);
- //upload_date = ' '.join(re.sub(r'[/,-]', r' ', mobj.group(1)).split())
- //upload_date = unified_strdate(upload_date)
- }
-
- // TODO: categories
-
- // description
- // TODO:
-
- // subtitles
- var videoSubtitles = null;
- queue(function () {
- return getSubtitles(video_id).done(function (subs) {
- videoSubtitles = subs;
- });
- });
- // TODO:
- //automatic_captions = self.extract_automatic_captions(video_id, video_webpage)
-
- var video_duration = null;
- if (!video_info.hasOwnProperty('length_seconds')) {
- return fail('unable to extract video duration');
- }
- else {
- video_duration = parseInt(decodeURIComponent(video_info['length_seconds'][0]));
- }
-
- // TODO:
- // annotations
- //video_annotations = None
- //if self._downloader.params.get('writeannotations', False):
- // video_annotations = self._extract_annotations(video_id)
-
- var formats = [];
-
- if (video_info.hasOwnProperty('conn') && video_info['conn'][0].startswith('rtmp')) {
- return fail('RTMP not supported');
- } else if (video_info.hasOwnProperty('url_encoded_fmt_stream_map') || video_info.hasOwnProperty('adaptive_fmts')) {
- var encodedUrlMap = '';
- if (video_info.hasOwnProperty('url_encoded_fmt_stream_map')) {
- encodedUrlMap = encodedUrlMap + ',' + video_info['url_encoded_fmt_stream_map'][0];
- }
- if (video_info.hasOwnProperty('adaptive_fmts')) {
- encodedUrlMap = encodedUrlMap + ',' + video_info['adaptive_fmts'][0];
- }
- if (encodedUrlMap.indexOf('rtmpe%3Dyes') !== -1) {
- return fail('rtmpe downloads are not supported');
- }
-
- var arr = encodedUrlMap.split(',');
- var size = arr.length;
-
- for (var i = 0; i < size; i++) {
- if (arr[i].length == 0) {
- continue;
- }
-
- var urlData = parseQS(arr[i]);
-
- if (!urlData.hasOwnProperty('itag') || !urlData.hasOwnProperty('url')) {
- continue;
- }
-
- var formatId = urlData['itag'][0];
- var url = urlData['url'][0];
-
- if (url.indexOf('ratebypass') === -1) {
- url += '&ratebypass=yes';
- }
-
- if (urlData.hasOwnProperty('sig')) {
- url += '&signature=' + urlData['sig'][0];
- formats.push({
- format_id: formatId,
- url: url
- });
- } else if (urlData.hasOwnProperty('s')) {
- var encrypted_sig = urlData['s'][0];
- var ASSETS_RE = /"assets":.+?"js":\s*("[^"]+")/;
-
- var jsplayer_url_json = searchRegex(ASSETS_RE, age_gate ? embed_webpage : video_webpage);
-
- // TODO:
- /*
- if not jsplayer_url_json and not age_gate:
- # We need the embed website after all
- if embed_webpage is None:
- embed_url = proto + '://www.youtube.com/embed/%s' % video_id
- embed_webpage = self._download(
- embed_url, video_id, 'Downloading embed webpage')
- jsplayer_url_json = self._searchRegex(
- ASSETS_RE, embed_webpage, 'JS player URL')
- */
-
- var playerUrl = JSON.parse(jsplayer_url_json);
-
- (function (encrypted_sig, playerUrl, formatId, url) {
- queue(function () {
- return decryptSignature(encrypted_sig, playerUrl).done(function (signature) {
- url += '&signature=' + signature;
- formats.push({
- format_id: formatId,
- url: url
- });
- });
- });
- })(encrypted_sig, playerUrl, formatId, url);
- } else if (url.indexOf('signature') !== -1) { // already decrypted
- formats.push({
- format_id: formatId,
- url: url
- });
- }
- }
- } else if (video_info.hasOwnProperty('hlsvp')) {
- return fail('HLS not supported');
- } else {
- return fail('no conn, hlsvp or url_encoded_fmt_stream_map information found in video info');
- }
-
- var dashManifestUrl = null;
-
- function buildResult(formats) {
- var size = formats.length;
-
- for (var i = 0; i < size; i++) {
- var fmt = formats[i];
- var master = YT_FORMATS[fmt['format_id']];
-
- $.extend(fmt, master);
- }
-
- return {
- 'id': video_id,
- 'uploader': video_uploader,
- 'uploader_id': video_uploader_id,
- 'upload_date': upload_date,
- 'title': video_title,
- 'thumbnail': 'https://i.ytimg.com/vi/' + video_id + '/hqdefault.jpg',
- //'description': video_description,
- //'categories': video_categories,
- subtitles: videoSubtitles,
- //'automatic_captions': automatic_captions,
- 'duration': video_duration,
- 'age_limit': age_gate ? 18 : 0,
- //'annotations': video_annotations,
- 'webpage_url': 'https://www.youtube.com/watch?v=' + video_id,
- 'view_count': view_count,
- //'average_rating': float_or_none(video_info.get('avg_rating', [None])[0]),
- 'formats': formats,
- dash_manifest_url: dashManifestUrl
- }
- }
-
- // Look for the DASH manifest
- if (video_info.hasOwnProperty('dashmpd')) {
- dashManifestUrl = video_info['dashmpd'][0];
-
- queue(function () {
- return parseDashManifest(video_id, dashManifestUrl, playerUrl, age_gate).done(function (dash) {
- if (dash != null) {
- dashManifestUrl = dash.dash_manifest_url;
- }
- deferred.resolve(buildResult(dash && dash.formats ? formats.concat(dash.formats) : formats));
- });
- });
- }
-
- return deferred.promise();
- }
-
- function extract(url) {
-
- var deferred = $.Deferred();
-
- var video_id = extractId(url);
-
- // Get video webpage
- url = 'https://www.youtube.com/watch?v=' + video_id + '&gl=US&hl=en&has_verified=1&bpctr=9999999999';
-
- download(url).done(function (video_webpage) {
- var age_gate = false;
-
- if (/player-age-gate-content">/i.test(video_webpage)) {
- age_gate = true;
- // We simulate the access to the video from www.youtube.com/v/{video_id}
- // this can be viewed without login into Youtube
- url = 'https://www.youtube.com/embed/' + video_id;
- download(url).done(function (embed_webpage) {
- var sts = searchRegex(/"sts"\s*:\s*(\d+)/, embed_webpage);
- var videoInfoUrl = 'https://www.youtube.com/get_video_info?video_id=' + video_id + '&eurl=' + encodeURIComponent('https://youtube.googleapis.com/v/' + video_id) + '&sts=' + sts;
- download(videoInfoUrl).done(function (video_info_webpage) {
- var video_info = parseQS(video_info_webpage);
- extractSupport(video_id, video_webpage, age_gate, embed_webpage, video_info).done(function (result) {
- deferred.resolve(result);
- });
- });
- });
- } else {
- age_gate = false;
- var videoInfoUrl = 'https://www.youtube.com/get_video_info?&video_id=' + video_id + '&el=detailpage&ps=default&eurl=&gl=US&hl=en'
- download(videoInfoUrl).done(function (video_info_webpage) {
- var videoInfo = parseQS(video_info_webpage);
- extractSupport(video_id, video_webpage, age_gate, '', videoInfo).done(function (result) {
- deferred.resolve(result);
- });
- }).fail(function () {
- deferred.resolve(null);
- });
- }
- });
-
- return deferred.promise();
- }
-
- function search(q) {
-
- var deferred = $.Deferred();
-
- var url = 'http://www.youtube.com/results?search_query=' + encodeURIComponent(q) + '&hl=en';
-
- download(url).done(function (html) {
- var re = /<h3 class="yt-lockup-title"><a href="\/watch\?v=(.*?)".*? title="(.*?)".*? Duration: (.*?)\.<\/span>.*?by <a href="\/user\/(.*?)".*?<li>([\d,]*) views<\/li>/ig;
- var m = null;
-
- var results = [];
-
- while (m = re.exec(html)) {
- var videoId = m[1];
-
- var r = {
- id: videoId,
- url: 'https://www.youtube.com/watch?v=' + videoId,
- title: m[2],
- duration: m[3],
- user: m[4],
- views: parseInt(m[5].replace(/,/g, ''), 10),
- thumbnail: 'https://i.ytimg.com/vi/' + videoId + '/mqdefault.jpg',
- };
-
- results.push(r);
- }
-
- deferred.resolve(results);
- });
-
- return deferred.promise();
- }
-
- function ytAutocompleteSource(request, response) {
-
- // setup global object
- if (typeof google === 'undefined') {
- google = {
- sbox: {
- p50: function (data) {
- data = data[1];
- var result = [];
- var size = data.length;
- for (var i = 0; i < size; i++) {
- result.push(data[i][0]);
- }
- google.sbox.response(result);
- },
- response: function (data) {
- log(data)
- }
- }
- }
- }
-
- // set global response
- google.sbox.response = response;
-
- $.ajax({
- url: 'https://clients1.google.com/complete/search?client=youtube&hl=en&gl=us&gs_rn=23&gs_ri=youtube&ds=yt&cp=2&gs_id=8',
- dataType: 'script',
- data: {
- q: request.term,
- callback: 'google.sbox.p50'
- },
- success: function (data) {
- // ignore
- }
- });
- }
- // END YT-LINKS CODE //
-
- // START WAITFORKEYELEMENTS CODE //
- /*--- waitForKeyElements(): A utility function, for Greasemonkey scripts,
- that detects and handles AJAXed content.
-
- Usage example:
-
- waitForKeyElements (
- "div.comments"
- , commentCallbackFunction
- );
-
- //--- Page-specific function to do what we want when the node is found.
- function commentCallbackFunction (jNode) {
- jNode.text ("This comment changed by waitForKeyElements().");
- }
-
- IMPORTANT: This function requires your script to have loaded jQuery.
- */
- function waitForKeyElements (
- selectorTxt, /* Required: The jQuery selector string that
- specifies the desired element(s).
- */
- actionFunction, /* Required: The code to run when elements are
- found. It is passed a jNode to the matched
- element.
- */
- bWaitOnce, /* Optional: If false, will continue to scan for
- new elements even after the first match is
- found.
- */
- iframeSelector /* Optional: If set, identifies the iframe to
- search.
- */
- ) {
- var targetNodes, btargetsFound;
-
- if (typeof iframeSelector == "undefined")
- targetNodes = $(selectorTxt);
- else
- targetNodes = $(iframeSelector).contents ()
- .find (selectorTxt);
-
- if (targetNodes && targetNodes.length > 0) {
- btargetsFound = true;
- /*--- Found target node(s). Go through each and act if they
- are new.
- */
- targetNodes.each ( function () {
- var jThis = $(this);
- var alreadyFound = jThis.data ('alreadyFound') || false;
-
- if (!alreadyFound) {
- //--- Call the payload function.
- var cancelFound = actionFunction (jThis);
- if (cancelFound)
- btargetsFound = false;
- else
- jThis.data ('alreadyFound', true);
- }
- } );
- }
- else {
- btargetsFound = false;
- }
-
- //--- Get the timer-control variable for this selector.
- var controlObj = waitForKeyElements.controlObj || {};
- var controlKey = selectorTxt.replace (/[^\w]/g, "_");
- var timeControl = controlObj [controlKey];
-
- //--- Now set or clear the timer as appropriate.
- if (btargetsFound && bWaitOnce && timeControl) {
- //--- The only condition where we need to clear the timer.
- clearInterval (timeControl);
- delete controlObj [controlKey]
- }
- else {
- //--- Set a timer, if needed.
- if ( ! timeControl) {
- timeControl = setInterval ( function () {
- waitForKeyElements ( selectorTxt,
- actionFunction,
- bWaitOnce,
- iframeSelector
- );
- },
- 300
- );
- controlObj [controlKey] = timeControl;
- }
- }
- waitForKeyElements.controlObj = controlObj;
- }
- // END WAITFORKEYELEMENTS CODE //
-