# -*- coding: utf-8 -*- import isodate import youtube_dl import logging import wx import config from googleapiclient.discovery import build from googleapiclient.errors import HttpError from update.utils import seconds_to_string from .import base DEVELOPER_KEY = "AIzaSyCU_hvZJEjLlAGAnlscquKEkE8l0lVOfn0" YOUTUBE_API_SERVICE_NAME = "youtube" YOUTUBE_API_VERSION = "v3" log = logging.getLogger("extractors.youtube.com") class interface(base.baseInterface): name = "YouTube" enabled = config.app["services"]["youtube"].get("enabled") def search(self, text, page=1): if text == "" or text == None: raise ValueError("Text must be passed and should not be blank.") if text.startswith("https") or text.startswith("http"): return self.search_from_url(text) type = "video" max_results = config.app["services"]["youtube"]["max_results"] log.debug("Retrieving data from Youtube...") youtube = build(YOUTUBE_API_SERVICE_NAME, YOUTUBE_API_VERSION, developerKey=DEVELOPER_KEY) search_response = youtube.search().list(q=text, part="id,snippet", maxResults=max_results, type=type).execute() self.results = [] ids = [] for search_result in search_response.get("items", []): if search_result["id"]["kind"] == "youtube#video": s = base.song(self) s.title = search_result["snippet"]["title"] ids.append(search_result["id"]["videoId"]) s.url = "https://www.youtube.com/watch?v="+search_result["id"]["videoId"] self.results.append(s) ssr = youtube.videos().list(id=",".join(ids), part="contentDetails", maxResults=1).execute() for i in range(len(self.results)): self.results[i].duration = seconds_to_string(isodate.parse_duration(ssr["items"][i]["contentDetails"]["duration"]).total_seconds()) log.debug("{0} results found.".format(len(self.results))) def search_from_url(self, url): log.debug("Getting download URL for {0}".format(url,)) if "playlist?list=" in url: return self.search_from_playlist(url) ydl = youtube_dl.YoutubeDL({'quiet': True, 'no_warnings': True, 'logger': log, 'prefer-free-formats': True, 'format': 'bestaudio', 'outtmpl': u'%(id)s%(ext)s'}) with ydl: result = ydl.extract_info(url, download=False) if 'entries' in result: videos = result['entries'] else: videos = [result] for video in videos: s = baseFile.song(self) s.title = video["title"] s.url = video["webpage_url"] # Cannot use direct URL here cause Youtube URLS expire after a minute. s.duration = seconds_to_string(video["duration"]) self.results.append(s) log.debug("{0} results found.".format(len(self.results))) def search_from_playlist(self, url): id = url.split("=")[1] max_results = 50 log.debug("Retrieving data from Youtube...") youtube = build(YOUTUBE_API_SERVICE_NAME, YOUTUBE_API_VERSION, developerKey=DEVELOPER_KEY) search_response = youtube.playlistItems().list(playlistId=id, part="id, status, snippet", maxResults=max_results).execute() self.results = [] ids = [] for search_result in search_response.get("items", []): if search_result["status"]["privacyStatus"] != "public": continue s = baseFile.song(self) s.title = search_result["snippet"]["title"] ids.append(search_result["snippet"]["resourceId"]["videoId"]) s.url = "https://www.youtube.com/watch?v="+search_result["snippet"]["resourceId"]["videoId"] self.results.append(s) ssr = youtube.videos().list(id=",".join(ids), part="contentDetails", maxResults=50).execute() for i in range(len(self.results)): self.results[i].duration = seconds_to_string(isodate.parse_duration(ssr["items"][i]["contentDetails"]["duration"]).total_seconds()) log.debug("{0} results found.".format(len(self.results))) def get_download_url(self, url): log.debug("Getting download URL for {0}".format(url,)) ydl = youtube_dl.YoutubeDL({'quiet': True, 'no_warnings': True, 'logger': log, 'format': 'bestaudio/best', 'outtmpl': u'%(id)s%(ext)s'}) with ydl: result = ydl.extract_info(url, download=False) if 'entries' in result: video = result['entries'][0] else: video = result # From here we should extract the first format so it will contain audio only. log.debug("Download URL: {0}".format(video["formats"][0]["url"],)) return video["formats"][0]["url"] def format_track(self, item): return "{0} {1}".format(item.title, item.duration) def transcoder_enabled(self): return config.app["services"]["youtube"]["transcode"] class settings(base.baseSettings): name = _("Youtube Settings") config_section = "youtube" def __init__(self, parent): super(settings, self).__init__(parent=parent) sizer = wx.BoxSizer(wx.VERTICAL) self.enabled = wx.CheckBox(self, wx.NewId(), _("Enable this service")) self.enabled.Bind(wx.EVT_CHECKBOX, self.on_enabled) self.map.append(("enabled", self.enabled)) sizer.Add(self.enabled, 0, wx.ALL, 5) max_results_label = wx.StaticText(self, wx.NewId(), _("Max results per page")) self.max_results = wx.SpinCtrl(self, wx.NewId()) self.max_results.SetRange(1, 50) max_results_sizer = wx.BoxSizer(wx.HORIZONTAL) max_results_sizer.Add(max_results_label, 0, wx.ALL, 5) max_results_sizer.Add(self.max_results, 0, wx.ALL, 5) self.map.append(("max_results", self.max_results)) # self.transcode = wx.CheckBox(self, wx.NewId(), _("Enable transcode when downloading")) # self.map.append(("transcode", self.transcode)) # sizer.Add(self.transcode, 0, wx.ALL, 5) self.SetSizer(sizer) def on_enabled(self, *args, **kwargs): for i in self.map: if i[1] != self.enabled: if self.enabled.GetValue() == True: i[1].Enable(True) else: i[1].Enable(False)