diff options
author | Stefano <l.stickell@yahoo.it> | 2013-11-25 12:21:50 +0100 |
---|---|---|
committer | Stefano <l.stickell@yahoo.it> | 2013-11-30 14:12:28 +0100 |
commit | cbbcc47c474c06cf0b1107698051e9f91fb9e9cb (patch) | |
tree | fe71317f200f930a34d16ed744a994414b6b63fe | |
parent | Merge pull request #397 from vuolter/s/hoster/BayfilesCom (diff) | |
download | pyload-cbbcc47c474c06cf0b1107698051e9f91fb9e9cb.tar.xz |
Merge pull request #335 from vuolter/s/crypter/YoutubeBatch
YoutubeBatch: Added support for download all videos available on a channel(cherry picked from commit 9f07c61c5758e67bdfdb88967258e4ae9e55abe0)
-rw-r--r-- | pyload/plugins/crypter/YoutubeBatch.py | 109 |
1 files changed, 90 insertions, 19 deletions
diff --git a/pyload/plugins/crypter/YoutubeBatch.py b/pyload/plugins/crypter/YoutubeBatch.py index b6178448d..ee84f0528 100644 --- a/pyload/plugins/crypter/YoutubeBatch.py +++ b/pyload/plugins/crypter/YoutubeBatch.py @@ -1,10 +1,28 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- +""" + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 3 of the License, + or (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + See the GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, see <http://www.gnu.org/licenses/>. + + @author: Walter Purcaro +""" + import re import json from module.plugins.Crypter import Crypter +from os.path import join API_KEY = "AIzaSyCKnWLNlkX-L4oD1aEzqqhRw1zczeD6_k0" @@ -12,33 +30,86 @@ API_KEY = "AIzaSyCKnWLNlkX-L4oD1aEzqqhRw1zczeD6_k0" class YoutubeBatch(Crypter): __name__ = "YoutubeBatch" __type__ = "container" - __pattern__ = r"https?://(?:[^/]*?)youtube\.com/(?:(?:view_play_list|playlist|.*?feature=PlayList).*?[?&](?:list|p)=)([a-zA-Z0-9-_]+)" - __version__ = "0.93" + __pattern__ = r"https?://(?:[^/]*?)youtube\.com/(?:(view_play_list|playlist|.*?feature=PlayList|user)(?:.*?[?&](?:list|p)=|/))([a-zA-Z0-9-_]+)" + __version__ = "0.94" __description__ = """Youtube.com Channel Download Plugin""" - __author_name__ = ("RaNaN", "Spoob", "zoidberg", "roland") - __author_mail__ = ("RaNaN@pyload.org", "spoob@pyload.org", "zoidberg@mujmail.cz", "roland@enkore.de") + __author_name__ = ("RaNaN", "Spoob", "zoidberg", "roland", "Walter Purcaro") + __author_mail__ = ("RaNaN@pyload.org", "spoob@pyload.org", "zoidberg@mujmail.cz", "roland@enkore.de", "vuolter@gmail.com") - def get_videos(self, playlist_id, token=None): - url = "https://www.googleapis.com/youtube/v3/playlistItems?playlistId=%s&part=snippet&key=%s&maxResults=50" % ( - playlist_id, API_KEY) + def json_response(self, api, req): + req.update({"key": API_KEY}) + url = "https://www.googleapis.com/youtube/v3/" + api + page = self.load(url, get=req) + return json.loads(page) + + def get_playlist_baseinfos(self, playlist_id): + res = self.json_response("playlists", {"part": "snippet", "id": playlist_id}) + + snippet = res["items"][0]["snippet"] + playlist_name = snippet["title"] + channel_title = snippet["channelTitle"] + return playlist_name, channel_title + + def get_channel_id(self, user_name): + res = self.json_response("channels", {"part": "id", "forUsername": user_name}) + return res["items"][0]["id"] + + def get_playlists(self, user_name, token=None): + channel_id = self.get_channel_id(user_name) + req = {"part": "id", "maxResults": "50", "channelId": channel_id} if token: - url += "&pageToken=" + token + req.update({"pageToken": token}) + res = self.json_response("playlists", req) - response = json.loads(self.load(url)) + for item in res["items"]: + yield item["id"] - for item in response["items"]: - if item["kind"] == "youtube#playlistItem" and item["snippet"]["resourceId"]["kind"] == "youtube#video": - yield "http://youtube.com/watch?v=" + item["snippet"]["resourceId"]["videoId"] + if "nextPageToken" in res: + for item in self.get_playlists(user_name, res["nextPageToken"]): + yield item - if "nextPageToken" in response: - for item in self.get_videos(playlist_id, response["nextPageToken"]): + def get_videos(self, playlist_id, token=None): + req = {"part": "snippet", "maxResults": "50", "playlistId": playlist_id} + if token: + req.update({"pageToken": token}) + res = self.json_response("playlistItems", req) + + for item in res["items"]: + yield "http://youtube.com/watch?v=" + item["snippet"]["resourceId"]["videoId"] + + if "nextPageToken" in res: + for item in self.get_videos(playlist_id, res["nextPageToken"]): yield item def decrypt(self, pyfile): - match_id = re.match(self.__pattern__, self.pyfile.url) - new_links = [] - playlist_id = match_id.group(1) + match_obj = re.match(self.__pattern__, pyfile.url) + match_type, match_result = match_obj.group(1), match_obj.group(2) + playlist_ids = [] + + #: is a channel username or just a playlist id? + if match_type == "user": + ids = self.get_playlists(match_result) + playlist_ids.extend(ids) + else: + playlist_ids.append(match_result) + + self.logDebug("Playlist IDs = %s" % playlist_ids) + + if not playlist_ids: + self.fail("Wrong url") + + for id in playlist_ids: + self.logDebug("Processing playlist id: %s" % id) + + playlist_name, channel_title = self.get_playlist_baseinfos(id) + video_links = [x for x in self.get_videos(id)] + + self.logInfo("%s videos found on playlist \"%s\" (channel \"%s\")" % (len(video_links), playlist_name, channel_title)) + + if not video_links: + continue - new_links.extend(self.get_videos(playlist_id)) + self.logDebug("Video links = %s" % video_links) - self.packages.append((self.pyfile.package().name, new_links, self.pyfile.package().name)) + folder = join(self.config['general']['download_folder'], channel_title, playlist_name) + self.packages.append((playlist_name, video_links, folder)) #Note: folder is NOT used actually! |