Hive - YouTube to Hive / Local Download

Inserts a download button on YouTube video pages and sends to hive -Major fixes

/** 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 //