# -*- coding: utf-8 -*- # # Test links: # http://ul.to/044yug9o # http://ul.to/gzfhd0xs import re from time import sleep from module.network.RequestFactory import getURL from module.plugins.Hoster import Hoster from module.plugins.Plugin import chunks from module.plugins.internal.CaptchaService import ReCaptcha from module.utils import html_unescape, parseFileSize key = "bGhGMkllZXByd2VEZnU5Y2NXbHhYVlZ5cEE1bkEzRUw=".decode('base64') def getID(url): """ returns id from file url""" m = re.match(UploadedTo.__pattern__, url) return m.group('ID') def getAPIData(urls): post = {"apikey": key} idMap = {} for i, url in enumerate(urls): id = getID(url) post['id_%s' % i] = id idMap[id] = url for _i in xrange(5): api = unicode(getURL("http://uploaded.net/api/filemultiple", post=post, decode=False), 'iso-8859-1') if api != "can't find request": break else: sleep(3) result = {} if api: for line in api.splitlines(): data = line.split(",", 4) if data[1] in idMap: result[data[1]] = (data[0], data[2], data[4], data[3], idMap[data[1]]) return result def parseFileInfo(self, url='', html=''): if not html and hasattr(self, "html"): html = self.html name = url size = 0 fileid = None if re.search(self.OFFLINE_PATTERN, html): # File offline status = 1 else: m = re.search(self.INFO_PATTERN, html) if m: name, fileid = html_unescape(m.group('N')), m.group('ID') size = parseFileSize(m.group('S')) status = 2 else: status = 3 return name, size, status, fileid def getInfo(urls): for chunk in chunks(urls, 80): result = [] api = getAPIData(chunk) for data in api.itervalues(): if data[0] == "online": result.append((html_unescape(data[2]), data[1], 2, data[4])) elif data[0] == "offline": result.append((data[4], 0, 1, data[4])) yield result class UploadedTo(Hoster): __name__ = "UploadedTo" __type__ = "hoster" __version__ = "0.75" __pattern__ = r'https?://(?:www\.)?(uploaded\.(to|net)|ul\.to)(/file/|/?\?id=|.*?&id=|/)(?P\w+)' __description__ = """Uploaded.net hoster plugin""" __license__ = "GPLv3" __authors__ = [("spoob", "spoob@pyload.org"), ("mkaay", "mkaay@mkaay.de"), ("zoidberg", "zoidberg@mujmail.cz"), ("netpok", "netpok@gmail.com"), ("stickell", "l.stickell@yahoo.it")] INFO_PATTERN = r'(?P[^<]+)  \s*]*>(?P[^<]+)' OFFLINE_PATTERN = r'Error: 404' DL_LIMIT_PATTERN = r'You have reached the max. number of possible free downloads for this hour' def setup(self): self.multiDL = self.resumeDownload = self.premium self.chunkLimit = 1 # critical problems with more chunks self.fileID = getID(self.pyfile.url) self.pyfile.url = "http://uploaded.net/file/%s" % self.fileID def process(self, pyfile): self.load("http://uploaded.net/language/en", just_header=True) api = getAPIData([pyfile.url]) # TODO: fallback to parse from site, because api sometimes delivers wrong status codes if not api: self.logWarning(_("No response for API call")) self.html = unicode(self.load(pyfile.url, decode=False), 'iso-8859-1') name, size, status, self.fileID = parseFileInfo(self) self.logDebug(name, size, status, self.fileID) if status == 1: self.offline() elif status == 2: pyfile.name, pyfile.size = name, size else: self.error(_("file info")) elif api == 'Access denied': self.fail(_("API key invalid")) else: if self.fileID not in api: self.offline() self.data = api[self.fileID] if self.data[0] != "online": self.offline() pyfile.name = html_unescape(self.data[2]) # pyfile.name = self.get_file_name() if self.premium: self.handlePremium() else: self.handleFree() def handlePremium(self): info = self.account.getAccountInfo(self.user, True) self.logDebug("%(name)s: Use Premium Account (%(left)sGB left)" % {"name": self.__name__, "left": info['trafficleft'] / 1024 / 1024}) if int(self.data[1]) / 1024 > info['trafficleft']: self.logInfo(_("Not enough traffic left")) self.account.empty(self.user) self.resetAccount() self.fail(_("Traffic exceeded")) header = self.load("http://uploaded.net/file/%s" % self.fileID, just_header=True) if 'location' in header: #Direct download self.logDebug("Direct download link detected") self.download(header['location']) else: #Indirect download self.html = self.load("http://uploaded.net/file/%s" % self.fileID) m = re.search(r'
(\d+) seconds", self.html) if m is None: self.fail(_("File not downloadable for free users")) self.setWait(int(m.group(1))) self.html = self.load("http://uploaded.net/js/download.js", decode=True) url = "http://uploaded.net/io/ticket/captcha/%s" % self.fileID downloadURL = "" recaptcha = ReCaptcha(self) for _i in xrange(5): challenge, response = recaptcha.challenge() options = {"recaptcha_challenge_field": challenge, "recaptcha_response_field": response} self.wait() result = self.load(url, post=options) self.logDebug("Result: %s" % result) if "limit-size" in result: self.fail(_("File too big for free download")) elif "limit-slot" in result: # Temporary restriction so just wait a bit self.setWait(30 * 60, True) self.wait() self.retry() elif "limit-parallel" in result: self.fail(_("Cannot download in parallel")) elif "limit-dl" in result or self.DL_LIMIT_PATTERN in result: # limit-dl self.setWait(3 * 60 * 60, True) self.wait() self.retry() elif '"err":"captcha"' in result: self.invalidCaptcha() elif "type:'download'" in result: self.correctCaptcha() downloadURL = re.search("url:'([^']+)", result).group(1) break else: self.error(_("Unknown error: %s") % result) if not downloadURL: self.fail(_("No Download url retrieved/all captcha attempts failed")) self.download(downloadURL, disposition=True) check = self.checkDownload({"limit-dl": self.DL_LIMIT_PATTERN}) if check == "limit-dl": self.setWait(3 * 60 * 60, True) self.wait() self.retry()