summaryrefslogtreecommitdiffstats
path: root/pyload
diff options
context:
space:
mode:
authorGravatar RaNaN <Mast3rRaNaN@hotmail.de> 2013-07-19 19:36:05 +0200
committerGravatar RaNaN <Mast3rRaNaN@hotmail.de> 2013-07-19 19:36:17 +0200
commit873f91be7d3316e93672731e2f3d2da02b41fca6 (patch)
tree5a12d1a5ac1f0a4aa5fca9ab9e6b86e64f25e388 /pyload
parentExplain how to add tips for translators (diff)
downloadpyload-873f91be7d3316e93672731e2f3d2da02b41fca6.tar.xz
new plugin type and refactored request classes
Diffstat (limited to 'pyload')
-rw-r--r--pyload/PluginManager.py2
-rw-r--r--pyload/datatypes/PyFile.py16
-rw-r--r--pyload/network/Browser.py2
-rw-r--r--pyload/network/CookieJar.py54
-rw-r--r--pyload/network/RequestFactory.py66
-rw-r--r--pyload/plugins/Account.py4
-rw-r--r--pyload/plugins/Crypter.py2
-rw-r--r--pyload/plugins/Download.py48
-rw-r--r--pyload/plugins/Hoster.py48
-rw-r--r--pyload/plugins/Request.py77
-rw-r--r--pyload/plugins/network/CurlChunk.py299
-rw-r--r--pyload/plugins/network/CurlDownload.py323
-rw-r--r--pyload/plugins/network/CurlRequest.py314
-rw-r--r--pyload/plugins/network/DefaultRequest.py9
14 files changed, 1156 insertions, 108 deletions
diff --git a/pyload/PluginManager.py b/pyload/PluginManager.py
index 149d04fc2..182768689 100644
--- a/pyload/PluginManager.py
+++ b/pyload/PluginManager.py
@@ -52,7 +52,7 @@ class BaseAttributes(defaultdict):
class PluginManager:
ROOT = "pyload.plugins."
LOCALROOT = "localplugins."
- TYPES = ("crypter", "hoster", "accounts", "addons", "internal")
+ TYPES = ("crypter", "hoster", "accounts", "addons", "network", "internal")
BUILTIN = re.compile(r'__(?P<attr>[a-z0-9_]+)__\s*=\s?(True|False|None|[0-9x.]+)', re.I)
SINGLE = re.compile(r'__(?P<attr>[a-z0-9_]+)__\s*=\s*(?:r|u|_)?((?:(?<!")"(?!")|\'|\().*(?:(?<!")"(?!")|\'|\)))',
diff --git a/pyload/datatypes/PyFile.py b/pyload/datatypes/PyFile.py
index ba5208795..7bb3a4e31 100644
--- a/pyload/datatypes/PyFile.py
+++ b/pyload/datatypes/PyFile.py
@@ -202,8 +202,10 @@ class PyFile(object):
sleep(0.1)
self.abort = False
- if self.plugin and self.plugin.req:
- self.plugin.req.abortDownloads()
+ if self.plugin:
+ self.plugin.req.abort()
+ if self.plugin.dl:
+ self.plugin.dl.abort()
self.release()
@@ -224,7 +226,7 @@ class PyFile(object):
def getSpeed(self):
""" calculates speed """
try:
- return self.plugin.req.speed
+ return self.plugin.dl.speed
except:
return 0
@@ -241,22 +243,22 @@ class PyFile(object):
def getBytesArrived(self):
""" gets bytes arrived """
try:
- return self.plugin.req.arrived
+ return self.plugin.dl.arrived
except:
return 0
def getBytesLeft(self):
""" gets bytes left """
try:
- return self.plugin.req.size - self.plugin.req.arrived
+ return self.plugin.dl.size - self.plugin.dl.arrived
except:
return 0
def getSize(self):
""" get size of download """
try:
- if self.plugin.req.size:
- return self.plugin.req.size
+ if self.plugin.dl.size:
+ return self.plugin.dl.size
else:
return self.size
except:
diff --git a/pyload/network/Browser.py b/pyload/network/Browser.py
index 25cbf669b..262adaebd 100644
--- a/pyload/network/Browser.py
+++ b/pyload/network/Browser.py
@@ -6,7 +6,7 @@ from logging import getLogger
from HTTPRequest import HTTPRequest
from HTTPDownload import HTTPDownload
-
+# @ Deprecated
class Browser(object):
__slots__ = ("log", "options", "bucket", "cj", "_size", "http", "dl")
diff --git a/pyload/network/CookieJar.py b/pyload/network/CookieJar.py
index ea2c43a9e..3d39c66b9 100644
--- a/pyload/network/CookieJar.py
+++ b/pyload/network/CookieJar.py
@@ -1,53 +1,19 @@
# -*- 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: mkaay, RaNaN
-"""
-
from time import time
+from Cookie import SimpleCookie
-# TODO: replace with simplecookie?
-class CookieJar():
- def __init__(self, pluginname):
- self.cookies = {}
- self.pluginname = pluginname
-
- def __repr__(self):
- return ("<CookieJar plugin=%s>\n\t" % self.pluginname) + "\n\t".join(self.cookies.values())
-
- def addCookies(self, clist):
- for c in clist:
- name = c.split("\t")[5]
- self.cookies[name] = c
-
- def getCookies(self):
- return self.cookies.values()
+class CookieJar(SimpleCookie):
def getCookie(self, name):
- if name in self.cookies:
- return self.cookies[name].split("\t")[6]
- else:
- return None
+ return self[name].value
- def setCookie(self, domain, name, value, path="/", exp=None):
+ def setCookie(self, domain, name, value, path="/", exp=None, secure="FALSE"):
if not exp: exp = time() + 3600 * 24 * 180
- # dot makes it valid on all subdomains
- s = ".%s TRUE %s FALSE %s %s %s" % (domain.strip("."), path, exp, name, value)
- self.cookies[name] = s
-
- def clear(self):
- self.cookies = {} \ No newline at end of file
+ self[name] = value
+ self[name]["domain"] = domain
+ self[name]["path"] = path
+ self[name]["expires"] = exp
+ if secure == "TRUE":
+ self[name]["secure"] = secure
diff --git a/pyload/network/RequestFactory.py b/pyload/network/RequestFactory.py
index 1581be9fc..e6015219f 100644
--- a/pyload/network/RequestFactory.py
+++ b/pyload/network/RequestFactory.py
@@ -17,44 +17,20 @@
@author: mkaay, RaNaN
"""
-from threading import Lock
-
-from Browser import Browser
from Bucket import Bucket
-from HTTPRequest import HTTPRequest
from CookieJar import CookieJar
+from pyload.plugins.network.DefaultRequest import DefaultRequest, DefaultDownload
+
class RequestFactory():
def __init__(self, core):
- self.lock = Lock()
self.core = core
self.bucket = Bucket()
self.updateBucket()
- @property
- def iface(self):
- return self.core.config["download"]["interface"]
-
- def getRequest(self, pluginName, cj=None):
- # TODO: mostly obsolete, browser could be removed completely
- req = Browser(self.bucket, self.getOptions())
-
- if cj:
- req.setCookieJar(cj)
- else:
- req.setCookieJar(CookieJar(pluginName))
-
- return req
-
- def getHTTPRequest(self, **kwargs):
- """ returns a http request, don't forget to close it ! """
- options = self.getOptions()
- options.update(kwargs) # submit kwargs as additional options
- return HTTPRequest(CookieJar(None), options)
-
def getURL(self, *args, **kwargs):
""" see HTTPRequest for argument list """
- h = HTTPRequest(None, self.getOptions())
+ h = DefaultRequest(self.getConfig())
try:
rep = h.load(*args, **kwargs)
finally:
@@ -62,9 +38,25 @@ class RequestFactory():
return rep
- def openCookieJar(self, pluginname):
- """Create new CookieJar"""
- return CookieJar(pluginname)
+ ########## old api methods above
+
+ def getRequest(self, context=None, klass=DefaultRequest):
+ """ Creates a request with new or given context """
+ # also accepts cookiejars, not so nice, since it depends on implementation
+ if isinstance(context, CookieJar):
+ return klass(self.getConfig(), context)
+ elif context:
+ return klass(*context)
+ else:
+ return klass(self.getConfig())
+
+ def getDownloadRequest(self, request=None, klass=DefaultDownload):
+ """ Instantiates a instance for downloading """
+ # TODO: load with plugin manager
+ return klass(self.bucket, request)
+
+ def getInterface(self):
+ return self.core.config["download"]["interface"]
def getProxies(self):
""" returns a proxy list for the request classes """
@@ -73,8 +65,10 @@ class RequestFactory():
else:
type = "http"
setting = self.core.config["proxy"]["type"].lower()
- if setting == "socks4": type = "socks4"
- elif setting == "socks5": type = "socks5"
+ if setting == "socks4":
+ type = "socks4"
+ elif setting == "socks5":
+ type = "socks5"
username = None
if self.core.config["proxy"]["username"] and self.core.config["proxy"]["username"].lower() != "none":
@@ -90,11 +84,11 @@ class RequestFactory():
"port": self.core.config["proxy"]["port"],
"username": username,
"password": pw,
- }
+ }
- def getOptions(self):
+ def getConfig(self):
"""returns options needed for pycurl"""
- return {"interface": self.iface,
+ return {"interface": self.getInterface(),
"proxies": self.getProxies(),
"ipv6": self.core.config["download"]["ipv6"]}
@@ -111,4 +105,4 @@ def getURL(*args, **kwargs):
def getRequest(*args, **kwargs):
- return pyreq.getHTTPRequest()
+ return pyreq.getRequest(*args, **kwargs)
diff --git a/pyload/plugins/Account.py b/pyload/plugins/Account.py
index 0bf201c1a..4492dfa18 100644
--- a/pyload/plugins/Account.py
+++ b/pyload/plugins/Account.py
@@ -70,7 +70,7 @@ class Account(Base):
self.lock = RLock()
self.timestamp = 0
self.login_ts = 0 # timestamp for login
- self.cj = CookieJar(self.__name__)
+ self.cj = CookieJar()
self.password = password
self.error = None
@@ -158,7 +158,7 @@ class Account(Base):
return self.options != before
def getAccountRequest(self):
- return self.core.requestFactory.getRequest(self.__name__, self.cj)
+ return self.core.requestFactory.getRequest(self.cj)
def getDownloadSettings(self):
""" Can be overwritten to change download settings. Default is no chunkLimit, max dl limit, resumeDownload
diff --git a/pyload/plugins/Crypter.py b/pyload/plugins/Crypter.py
index b881510ea..1401d68b8 100644
--- a/pyload/plugins/Crypter.py
+++ b/pyload/plugins/Crypter.py
@@ -102,7 +102,7 @@ class Crypter(Base):
def __init__(self, core, package=None, password=None):
Base.__init__(self, core)
- self.req = core.requestFactory.getRequest(self.__name__)
+ self.req = core.requestFactory.getRequest()
# Package the plugin was initialized for, don't use this, its not guaranteed to be set
self.package = package
diff --git a/pyload/plugins/Download.py b/pyload/plugins/Download.py
new file mode 100644
index 000000000..e86089dc3
--- /dev/null
+++ b/pyload/plugins/Download.py
@@ -0,0 +1,48 @@
+# -*- coding: utf-8 -*-
+
+from Request import Request
+
+class Download(Request):
+ """ Abstract class for download request """
+
+ __version__ = "0.1"
+
+ def __init__(self, bucket, request=None):
+ # Copies the context
+ context = request.getContext() if request else [{}]
+ Request.__init__(self, *context)
+
+ self._running = False
+ self._name = None
+ self._size = 0
+
+ #: bucket used for rate limiting
+ self.bucket = bucket
+
+ def download(self, uri, path, *args, **kwargs):
+ """ Downloads the resource with additional options depending on implementation """
+ raise NotImplementedError
+
+ @property
+ def running(self):
+ return self._running
+
+ @property
+ def size(self):
+ """ Size in bytes """
+ return self._size
+
+ @property
+ def name(self):
+ """ Name of the resource if known """
+ return self._name
+
+ @property
+ def speed(self):
+ """ Download rate in bytes per second """
+ return 0
+
+ @property
+ def arrived(self):
+ """ Number of bytes already loaded """
+ return 0 \ No newline at end of file
diff --git a/pyload/plugins/Hoster.py b/pyload/plugins/Hoster.py
index bc82aac09..a37102aff 100644
--- a/pyload/plugins/Hoster.py
+++ b/pyload/plugins/Hoster.py
@@ -25,25 +25,36 @@ if os.name != "nt":
from pwd import getpwnam
from grp import getgrnam
-from Base import Base, Fail, Retry
from pyload.utils import chunks as _chunks
-from pyload.utils.fs import save_join, save_filename, fs_encode, fs_decode,\
+from pyload.utils.fs import save_join, save_filename, fs_encode, fs_decode, \
remove, makedirs, chmod, stat, exists, join
+from Base import Base, Fail, Retry
+from network.DefaultRequest import DefaultRequest, DefaultDownload
+
# Import for Hoster Plugins
chunks = _chunks
+
class Reconnect(Exception):
""" raised when reconnected """
+
class SkipDownload(Exception):
""" raised when download should be skipped """
+
class Hoster(Base):
"""
Base plugin for hoster plugin. Overwrite getInfo for online status retrieval, process for downloading.
"""
+ #: Class used to make requests with `self.load`
+ REQUEST_CLASS = DefaultRequest
+
+ #: Class used to make download
+ DOWNLOAD_CLASS = DefaultDownload
+
@staticmethod
def getInfo(urls):
"""This method is used to retrieve the online status of files for hoster plugins.
@@ -87,7 +98,10 @@ class Hoster(Base):
self.chunkLimit, self.limitDL, self.resumeDownload = self.account.getDownloadSettings()
self.premium = self.account.isPremium()
else:
- self.req = self.core.requestFactory.getRequest(self.__name__)
+ self.req = self.core.requestFactory.getRequest(klass=self.REQUEST_CLASS)
+
+ #: Will hold the download class
+ self.dl = None
#: associated pyfile instance, see `PyFile`
self.pyfile = pyfile
@@ -150,7 +164,7 @@ class Hoster(Base):
# will force a re-login or reload of account info if necessary
self.account.getAccountInfo()
else:
- self.req.clearCookies()
+ self.req.reset()
self.setup()
@@ -242,11 +256,6 @@ class Hoster(Base):
def download(self, url, get={}, post={}, ref=True, cookies=True, disposition=False):
"""Downloads the content at url to download folder
- :param url:
- :param get:
- :param post:
- :param ref:
- :param cookies:
:param disposition: if True and server provides content-disposition header\
the filename will be changed if needed
:return: The location where the file was saved
@@ -280,12 +289,15 @@ class Hoster(Base):
self.core.addonManager.dispatchEvent("download:start", self.pyfile, url, filename)
+ # Create the class used for downloading
+ self.dl = self.core.requestFactory.getDownloadRequest(self.req, self.DOWNLOAD_CLASS)
try:
- newname = self.req.httpDownload(url, filename, get=get, post=post, ref=ref, cookies=cookies,
- chunks=self.getChunkCount(), resume=self.resumeDownload,
- disposition=disposition)
+ # TODO: hardcoded arguments
+ newname = self.dl.download(url, filename, get=get, post=post, referer=ref, chunks=self.getChunkCount(),
+ resume=self.resumeDownload, disposition=disposition)
finally:
- self.pyfile.size = self.req.size
+ self.dl.close()
+ self.pyfile.size = self.dl.size
if disposition and newname and newname != name: #triple check, just to be sure
self.log.info("%(name)s saved as %(newname)s" % {"name": name, "newname": newname})
@@ -325,8 +337,10 @@ class Hoster(Base):
size = stat(lastDownload)
size = size.st_size
- if api_size and api_size <= size: return None
- elif size > max_size and not read_size: return None
+ if api_size and api_size <= size:
+ return None
+ elif size > max_size and not read_size:
+ return None
self.log.debug("Download Check triggered")
f = open(lastDownload, "rb")
content = f.read(read_size if read_size else -1)
@@ -369,7 +383,7 @@ class Hoster(Base):
if pyfile.status in (0, 12): #finished or downloading
raise SkipDownload(pyfile.pluginname)
elif pyfile.status in (
- 5, 7) and starting: #a download is waiting/starting and was apparently started before
+ 5, 7) and starting: #a download is waiting/starting and was apparently started before
raise SkipDownload(pyfile.pluginname)
download_folder = self.config['general']['download_folder']
@@ -394,6 +408,8 @@ class Hoster(Base):
if hasattr(self, "req"):
self.req.close()
del self.req
+ if hasattr(self, "dl"):
+ del self.dl
if hasattr(self, "thread"):
del self.thread
if hasattr(self, "html"):
diff --git a/pyload/plugins/Request.py b/pyload/plugins/Request.py
new file mode 100644
index 000000000..8e8e0cc6b
--- /dev/null
+++ b/pyload/plugins/Request.py
@@ -0,0 +1,77 @@
+# -*- coding: utf-8 -*-
+
+from logging import getLogger
+
+
+class ResponseException(Exception):
+ def __init__(self, code, content=""):
+ Exception.__init__(self, "Server response error: %s %s" % (code, content))
+ self.code = code
+
+class Request(object):
+ """ Abstract class to support different types of request, most methods should be overwritten """
+
+ __version__ = "0.1"
+
+ #: Class that will be instantiated and associated with the request, and if needed copied and reused
+ CONTEXT_CLASS = None
+
+ def __init__(self, config, context=None, options=None):
+ self.log = getLogger("log")
+
+ # Global config, holds some configurable parameter
+ self.config = config
+
+ # Create a new context if not given
+ if context is None and self.CONTEXT_CLASS is not None:
+ self.context = self.CONTEXT_CLASS()
+ else:
+ self.context = context
+
+ # Store options in dict
+ self.options = {} if options is None else options
+
+ # Last response code
+ self.code = 0
+ self.doAbort = False
+ self.initContext()
+
+ # TODO: content encoding? Could be handled globally
+
+ def initContext(self):
+ """ Should be used to initialize everything from given context and options """
+ pass
+
+ def getContext(self):
+ """ Retrieves complete state that is needed to copy the request context """
+ return self.config, self.context, self.options
+
+ def setContext(self, *args):
+ """ Sets request context """
+ self.config, self.context, self.options = args
+
+ def setOption(self, name, value):
+ """ Sets an option """
+ self.options[name] = value
+
+ def unsetOption(self, name):
+ """ Removes a specific option or reset everything on empty string """
+ if name == "":
+ self.options.clear()
+ else:
+ del self.options[name]
+
+ def load(self, uri, *args, **kwargs):
+ """ Loads given resource from given uri. Args and kwargs depends on implementation"""
+ raise NotImplementedError
+
+ def abort(self):
+ self.doAbort = True
+
+ def reset(self):
+ """ Resets the context to initial state """
+ self.unsetOption("")
+
+ def close(self):
+ """ Close and clean everything """
+ pass \ No newline at end of file
diff --git a/pyload/plugins/network/CurlChunk.py b/pyload/plugins/network/CurlChunk.py
new file mode 100644
index 000000000..4250db2ce
--- /dev/null
+++ b/pyload/plugins/network/CurlChunk.py
@@ -0,0 +1,299 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+###############################################################################
+# Copyright(c) 2008-2012 pyLoad Team
+# http://www.pyload.org
+#
+# This file is part of pyLoad.
+# pyLoad is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as
+# published by the Free Software Foundation, either version 3 of the
+# License, or (at your option) any later version.
+#
+# Subjected to the terms and conditions in LICENSE
+#
+# @author: RaNaN
+###############################################################################
+
+from os import remove, stat, fsync
+from os.path import exists
+from time import sleep
+from re import search
+
+import codecs
+import pycurl
+
+from pyload.utils import remove_chars
+from pyload.utils.fs import fs_encode
+
+from CurlRequest import CurlRequest
+
+class WrongFormat(Exception):
+ pass
+
+
+class ChunkInfo():
+ def __init__(self, name):
+ self.name = unicode(name)
+ self.size = 0
+ self.resume = False
+ self.chunks = []
+
+ def __repr__(self):
+ ret = "ChunkInfo: %s, %s\n" % (self.name, self.size)
+ for i, c in enumerate(self.chunks):
+ ret += "%s# %s\n" % (i, c[1])
+
+ return ret
+
+ def setSize(self, size):
+ self.size = int(size)
+
+ def addChunk(self, name, range):
+ self.chunks.append((name, range))
+
+ def clear(self):
+ self.chunks = []
+
+ def createChunks(self, chunks):
+ self.clear()
+ chunk_size = self.size / chunks
+
+ current = 0
+ for i in range(chunks):
+ end = self.size - 1 if (i == chunks - 1) else current + chunk_size
+ self.addChunk("%s.chunk%s" % (self.name, i), (current, end))
+ current += chunk_size + 1
+
+
+ def save(self):
+ fs_name = fs_encode("%s.chunks" % self.name)
+ fh = codecs.open(fs_name, "w", "utf_8")
+ fh.write("name:%s\n" % self.name)
+ fh.write("size:%s\n" % self.size)
+ for i, c in enumerate(self.chunks):
+ fh.write("#%d:\n" % i)
+ fh.write("\tname:%s\n" % c[0])
+ fh.write("\trange:%i-%i\n" % c[1])
+ fh.close()
+
+ @staticmethod
+ def load(name):
+ fs_name = fs_encode("%s.chunks" % name)
+ if not exists(fs_name):
+ raise IOError()
+ fh = codecs.open(fs_name, "r", "utf_8")
+ name = fh.readline()[:-1]
+ size = fh.readline()[:-1]
+ if name.startswith("name:") and size.startswith("size:"):
+ name = name[5:]
+ size = size[5:]
+ else:
+ fh.close()
+ raise WrongFormat()
+ ci = ChunkInfo(name)
+ ci.loaded = True
+ ci.setSize(size)
+ while True:
+ if not fh.readline(): #skip line
+ break
+ name = fh.readline()[1:-1]
+ range = fh.readline()[1:-1]
+ if name.startswith("name:") and range.startswith("range:"):
+ name = name[5:]
+ range = range[6:].split("-")
+ else:
+ raise WrongFormat()
+
+ ci.addChunk(name, (long(range[0]), long(range[1])))
+ fh.close()
+ return ci
+
+ def remove(self):
+ fs_name = fs_encode("%s.chunks" % self.name)
+ if exists(fs_name): remove(fs_name)
+
+ def getCount(self):
+ return len(self.chunks)
+
+ def getChunkName(self, index):
+ return self.chunks[index][0]
+
+ def getChunkRange(self, index):
+ return self.chunks[index][1]
+
+
+class CurlChunk(CurlRequest):
+ def __init__(self, id, parent, range=None, resume=False):
+ self.setContext(*parent.getContext())
+
+ self.id = id
+ self.p = parent # CurlDownload instance
+ self.range = range # tuple (start, end)
+ self.resume = resume
+ self.log = parent.log
+
+ self.size = range[1] - range[0] if range else -1
+ self.arrived = 0
+ self.lastURL = self.p.referer
+
+ self.c = pycurl.Curl()
+
+ self.header = ""
+ self.headerParsed = False #indicates if the header has been processed
+
+ self.fp = None #file handle
+
+ self.initContext()
+
+ self.BOMChecked = False # check and remove byte order mark
+
+ self.rep = None
+
+ self.sleep = 0.000
+ self.lastSize = 0
+
+ def __repr__(self):
+ return "<CurlChunk id=%d, size=%d, arrived=%d>" % (self.id, self.size, self.arrived)
+
+ @property
+ def cj(self):
+ return self.p.context
+
+ def getHandle(self):
+ """ returns a Curl handle ready to use for perform/multiperform """
+
+ self.setRequestContext(self.p.url, self.p.get, self.p.post, self.p.referer, self.cj)
+ self.c.setopt(pycurl.WRITEFUNCTION, self.writeBody)
+ self.c.setopt(pycurl.HEADERFUNCTION, self.writeHeader)
+
+ # request all bytes, since some servers in russia seems to have a defect arihmetic unit
+
+ fs_name = fs_encode(self.p.info.getChunkName(self.id))
+ if self.resume:
+ self.fp = open(fs_name, "ab")
+ self.arrived = self.fp.tell()
+ if not self.arrived:
+ self.arrived = stat(fs_name).st_size
+
+ if self.range:
+ #do nothing if chunk already finished
+ if self.arrived + self.range[0] >= self.range[1]: return None
+
+ if self.id == len(self.p.info.chunks) - 1: #as last chunk dont set end range, so we get everything
+ range = "%i-" % (self.arrived + self.range[0])
+ else:
+ range = "%i-%i" % (self.arrived + self.range[0], min(self.range[1] + 1, self.p.size - 1))
+
+ self.log.debug("Chunked resume with range %s" % range)
+ self.c.setopt(pycurl.RANGE, range)
+ else:
+ self.log.debug("Resume File from %i" % self.arrived)
+ self.c.setopt(pycurl.RESUME_FROM, self.arrived)
+
+ else:
+ if self.range:
+ if self.id == len(self.p.info.chunks) - 1: # see above
+ range = "%i-" % self.range[0]
+ else:
+ range = "%i-%i" % (self.range[0], min(self.range[1] + 1, self.p.size - 1))
+
+ self.log.debug("Chunked with range %s" % range)
+ self.c.setopt(pycurl.RANGE, range)
+
+ self.fp = open(fs_name, "wb")
+
+ return self.c
+
+ def writeHeader(self, buf):
+ self.header += buf
+ #@TODO forward headers?, this is possibly unneeded, when we just parse valid 200 headers
+ # as first chunk, we will parse the headers
+ if not self.range and self.header.endswith("\r\n\r\n"):
+ self.parseHeader()
+ elif not self.range and buf.startswith("150") and "data connection" in buf: #ftp file size parsing
+ size = search(r"(\d+) bytes", buf)
+ if size:
+ self.p._size = int(size.group(1))
+ self.p.chunkSupport = True
+
+ self.headerParsed = True
+
+ def writeBody(self, buf):
+ #ignore BOM, it confuses unrar
+ if not self.BOMChecked:
+ if [ord(b) for b in buf[:3]] == [239, 187, 191]:
+ buf = buf[3:]
+ self.BOMChecked = True
+
+ size = len(buf)
+
+ self.arrived += size
+
+ self.fp.write(buf)
+
+ if self.p.bucket:
+ sleep(self.p.bucket.consumed(size))
+ else:
+ # Avoid small buffers, increasing sleep time slowly if buffer size gets smaller
+ # otherwise reduce sleep time percentile (values are based on tests)
+ # So in general cpu time is saved without reducing bandwidth too much
+
+ if size < self.lastSize:
+ self.sleep += 0.002
+ else:
+ self.sleep *= 0.7
+
+ self.lastSize = size
+
+ sleep(self.sleep)
+
+ if self.range and self.arrived > self.size:
+ return 0 #close if we have enough data
+
+
+ def parseHeader(self):
+ """parse data from received header"""
+ for orgline in self.decodeResponse(self.header).splitlines():
+ line = orgline.strip().lower()
+ if line.startswith("accept-ranges") and "bytes" in line:
+ self.p.chunkSupport = True
+
+ if "content-disposition" in line:
+
+ m = search("filename(?P<type>=|\*=(?P<enc>.+)'')(?P<name>.*)", line)
+ if m:
+ name = remove_chars(m.groupdict()['name'], "\"';/").strip()
+ self.p._name = name
+ self.log.debug("Content-Disposition: %s" % name)
+
+ if not self.resume and line.startswith("content-length"):
+ self.p._size = int(line.split(":")[1])
+
+ self.headerParsed = True
+
+ def stop(self):
+ """The download will not proceed after next call of writeBody"""
+ self.range = [0,0]
+ self.size = 0
+
+ def resetRange(self):
+ """ Reset the range, so the download will load all data available """
+ self.range = None
+
+ def setRange(self, range):
+ self.range = range
+ self.size = range[1] - range[0]
+
+ def flushFile(self):
+ """ flush and close file """
+ self.fp.flush()
+ fsync(self.fp.fileno()) #make sure everything was written to disk
+ self.fp.close() #needs to be closed, or merging chunks will fail
+
+ def close(self):
+ """ closes everything, unusable after this """
+ if self.fp: self.fp.close()
+ self.c.close()
+ if hasattr(self, "p"): del self.p
diff --git a/pyload/plugins/network/CurlDownload.py b/pyload/plugins/network/CurlDownload.py
new file mode 100644
index 000000000..5de83ec7b
--- /dev/null
+++ b/pyload/plugins/network/CurlDownload.py
@@ -0,0 +1,323 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+###############################################################################
+# Copyright(c) 2008-2012 pyLoad Team
+# http://www.pyload.org
+#
+# This file is part of pyLoad.
+# pyLoad is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as
+# published by the Free Software Foundation, either version 3 of the
+# License, or (at your option) any later version.
+#
+# Subjected to the terms and conditions in LICENSE
+#
+# @author: RaNaN
+###############################################################################
+
+from os import remove
+from os.path import dirname
+from time import time
+from shutil import move
+
+import pycurl
+
+from pyload.plugins.Base import Abort
+from pyload.utils.fs import save_join, fs_encode
+
+from ..Download import Download
+from CurlChunk import ChunkInfo, CurlChunk
+from CurlRequest import ResponseException
+
+# TODO: save content-disposition for resuming
+
+class CurlDownload(Download):
+ """ loads an url, http + ftp supported """
+
+ # def __init__(self, url, filename, get={}, post={}, referer=None, cj=None, bucket=None,
+ # options={}, disposition=False):
+
+ def __init__(self, *args, **kwargs):
+ Download.__init__(self, *args, **kwargs)
+
+ self.path = None
+ self.disposition = False
+
+ self.chunks = []
+ self.chunkSupport = None
+
+ self.m = pycurl.CurlMulti()
+
+ #needed for speed calculation
+ self.lastArrived = []
+ self.speeds = []
+ self.lastSpeeds = [0, 0]
+
+ @property
+ def speed(self):
+ last = [sum(x) for x in self.lastSpeeds if x]
+ return (sum(self.speeds) + sum(last)) / (1 + len(last))
+
+ @property
+ def arrived(self):
+ return sum(c.arrived for c in self.chunks) if self.chunks else self._size
+
+ @property
+ def name(self):
+ return self._name if self.disposition else None
+
+ def _copyChunks(self):
+ init = fs_encode(self.info.getChunkName(0)) #initial chunk name
+
+ if self.info.getCount() > 1:
+ fo = open(init, "rb+") #first chunkfile
+ for i in range(1, self.info.getCount()):
+ #input file
+ fo.seek(
+ self.info.getChunkRange(i - 1)[1] + 1) #seek to beginning of chunk, to get rid of overlapping chunks
+ fname = fs_encode("%s.chunk%d" % (self.path, i))
+ fi = open(fname, "rb")
+ buf = 32 * 1024
+ while True: #copy in chunks, consumes less memory
+ data = fi.read(buf)
+ if not data:
+ break
+ fo.write(data)
+ fi.close()
+ if fo.tell() < self.info.getChunkRange(i)[1]:
+ fo.close()
+ remove(init)
+ self.info.remove() #there are probably invalid chunks
+ raise Exception("Downloaded content was smaller than expected. Try to reduce download connections.")
+ remove(fname) #remove chunk
+ fo.close()
+
+ if self.name:
+ self.filename = save_join(dirname(self.path), self.name)
+
+ move(init, fs_encode(self.path))
+ self.info.remove() #remove info file
+
+ def checkResume(self):
+ try:
+ self.info = ChunkInfo.load(self.path)
+ self.info.resume = True #resume is only possible with valid info file
+ self._size = self.info.size
+ self.infoSaved = True
+ except IOError:
+ self.info = ChunkInfo(self.path)
+
+ def download(self, uri, path, get={}, post={}, referer=True, disposition=False, chunks=1, resume=False):
+ """ returns new filename or None """
+ self.url = uri
+ self.path = path
+ self.disposition = disposition
+ self.get = get
+ self.post = post
+ self.referer = referer
+
+ self.checkResume()
+ chunks = max(1, chunks)
+ resume = self.info.resume and resume
+
+ try:
+ self._download(chunks, resume)
+ except pycurl.error, e:
+ #code 33 - no resume
+ code = e.args[0]
+ if code == 33:
+ # try again without resume
+ self.log.debug("Errno 33 -> Restart without resume")
+
+ #remove old handles
+ for chunk in self.chunks:
+ self.closeChunk(chunk)
+
+ return self._download(chunks, False)
+ else:
+ raise
+ finally:
+ self.close()
+
+ return self.name
+
+ def _download(self, chunks, resume):
+ if not resume:
+ self.info.clear()
+ self.info.addChunk("%s.chunk0" % self.path, (0, 0)) #create an initial entry
+
+ self.chunks = []
+
+ init = CurlChunk(0, self, None, resume) #initial chunk that will load complete file (if needed)
+
+ self.chunks.append(init)
+ self.m.add_handle(init.getHandle())
+
+ lastFinishCheck = 0
+ lastTimeCheck = 0
+ chunksDone = set() # list of curl handles that are finished
+ chunksCreated = False
+ done = False
+ if self.info.getCount() > 1: # This is a resume, if we were chunked originally assume still can
+ self.chunkSupport = True
+
+ while 1:
+ #need to create chunks
+ if not chunksCreated and self.chunkSupport and self.size: #will be set later by first chunk
+
+ if not resume:
+ self.info.setSize(self.size)
+ self.info.createChunks(chunks)
+ self.info.save()
+
+ chunks = self.info.getCount()
+
+ init.setRange(self.info.getChunkRange(0))
+
+ for i in range(1, chunks):
+ c = CurlChunk(i, self, self.info.getChunkRange(i), resume)
+
+ handle = c.getHandle()
+ if handle:
+ self.chunks.append(c)
+ self.m.add_handle(handle)
+ else:
+ #close immediately
+ self.log.debug("Invalid curl handle -> closed")
+ c.close()
+
+ chunksCreated = True
+
+ while 1:
+ ret, num_handles = self.m.perform()
+ if ret != pycurl.E_CALL_MULTI_PERFORM:
+ break
+
+ t = time()
+
+ # reduce these calls
+ # when num_q is 0, the loop is exited
+ while lastFinishCheck + 0.5 < t:
+ # list of failed curl handles
+ failed = []
+ ex = None # save only last exception, we can only raise one anyway
+
+ num_q, ok_list, err_list = self.m.info_read()
+ for c in ok_list:
+ chunk = self.findChunk(c)
+ try: # check if the header implies success, else add it to failed list
+ chunk.verifyHeader()
+ except ResponseException, e:
+ self.log.debug("Chunk %d failed: %s" % (chunk.id + 1, str(e)))
+ failed.append(chunk)
+ ex = e
+ else:
+ chunksDone.add(c)
+
+ for c in err_list:
+ curl, errno, msg = c
+ chunk = self.findChunk(curl)
+ #test if chunk was finished
+ if errno != 23 or "0 !=" not in msg:
+ failed.append(chunk)
+ ex = pycurl.error(errno, msg)
+ self.log.debug("Chunk %d failed: %s" % (chunk.id + 1, str(ex)))
+ continue
+
+ try: # check if the header implies success, else add it to failed list
+ chunk.verifyHeader()
+ except ResponseException, e:
+ self.log.debug("Chunk %d failed: %s" % (chunk.id + 1, str(e)))
+ failed.append(chunk)
+ ex = e
+ else:
+ chunksDone.add(curl)
+ if not num_q: # no more info to get
+
+ # check if init is not finished so we reset download connections
+ # note that other chunks are closed and everything downloaded with initial connection
+ if failed and init not in failed and init.c not in chunksDone:
+ self.log.error(_("Download chunks failed, fallback to single connection | %s" % (str(ex))))
+
+ #list of chunks to clean and remove
+ to_clean = filter(lambda x: x is not init, self.chunks)
+ for chunk in to_clean:
+ self.closeChunk(chunk)
+ self.chunks.remove(chunk)
+ remove(fs_encode(self.info.getChunkName(chunk.id)))
+
+ #let first chunk load the rest and update the info file
+ init.resetRange()
+ self.info.clear()
+ self.info.addChunk("%s.chunk0" % self.filename, (0, self.size))
+ self.info.save()
+ elif failed:
+ raise ex
+
+ lastFinishCheck = t
+
+ if len(chunksDone) >= len(self.chunks):
+ if len(chunksDone) > len(self.chunks):
+ self.log.warning("Finished download chunks size incorrect, please report bug.")
+ done = True #all chunks loaded
+
+ break
+
+ if done:
+ break #all chunks loaded
+
+ # calc speed once per second, averaging over 3 seconds
+ if lastTimeCheck + 1 < t:
+ diff = [c.arrived - (self.lastArrived[i] if len(self.lastArrived) > i else 0) for i, c in
+ enumerate(self.chunks)]
+
+ self.lastSpeeds[1] = self.lastSpeeds[0]
+ self.lastSpeeds[0] = self.speeds
+ self.speeds = [float(a) / (t - lastTimeCheck) for a in diff]
+ self.lastArrived = [c.arrived for c in self.chunks]
+ lastTimeCheck = t
+
+ if self.doAbort:
+ raise Abort()
+
+ self.m.select(1)
+
+ for chunk in self.chunks:
+ chunk.flushFile() #make sure downloads are written to disk
+
+ self._copyChunks()
+
+ def findChunk(self, handle):
+ """ linear search to find a chunk (should be ok since chunk size is usually low) """
+ for chunk in self.chunks:
+ if chunk.c == handle: return chunk
+
+ def closeChunk(self, chunk):
+ try:
+ self.m.remove_handle(chunk.c)
+ except pycurl.error, e:
+ self.log.debug("Error removing chunk: %s" % str(e))
+ finally:
+ chunk.close()
+
+ def close(self):
+ """ cleanup """
+ for chunk in self.chunks:
+ self.closeChunk(chunk)
+ else:
+ #Workaround: pycurl segfaults when closing multi, that never had any curl handles
+ if hasattr(self, "m"):
+ c = pycurl.Curl()
+ self.m.add_handle(c)
+ self.m.remove_handle(c)
+ c.close()
+
+ self.chunks = []
+ if hasattr(self, "m"):
+ self.m.close()
+ del self.m
+ if hasattr(self, "cj"):
+ del self.cj
+ if hasattr(self, "info"):
+ del self.info \ No newline at end of file
diff --git a/pyload/plugins/network/CurlRequest.py b/pyload/plugins/network/CurlRequest.py
new file mode 100644
index 000000000..4630403df
--- /dev/null
+++ b/pyload/plugins/network/CurlRequest.py
@@ -0,0 +1,314 @@
+# -*- coding: utf-8 -*-
+
+###############################################################################
+# Copyright(c) 2008-2012 pyLoad Team
+# http://www.pyload.org
+#
+# This file is part of pyLoad.
+# pyLoad is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as
+# published by the Free Software Foundation, either version 3 of the
+# License, or (at your option) any later version.
+#
+# Subjected to the terms and conditions in LICENSE
+#
+# @author: RaNaN
+###############################################################################
+
+import pycurl
+
+from codecs import getincrementaldecoder, lookup, BOM_UTF8
+from urllib import quote, urlencode
+from httplib import responses
+from cStringIO import StringIO
+
+from pyload.plugins.Base import Abort
+from pyload.network.CookieJar import CookieJar
+
+from ..Request import Request, ResponseException
+
+
+def myquote(url):
+ return quote(url.encode('utf8') if isinstance(url, unicode) else url, safe="%/:=&?~#+!$,;'@()*[]")
+
+
+def myurlencode(data):
+ data = dict(data)
+ return urlencode(dict((x.encode('utf8') if isinstance(x, unicode) else x, \
+ y.encode('utf8') if isinstance(y, unicode) else y ) for x, y in data.iteritems()))
+
+
+bad_headers = range(400, 418) + range(500, 506)
+
+
+class CurlRequest(Request):
+ """ Request class based on libcurl """
+
+ __version__ = "0.1"
+
+ CONTEXT_CLASS = CookieJar
+
+ def __init__(self, *args, **kwargs):
+ self.c = pycurl.Curl()
+ Request.__init__(self, *args, **kwargs)
+
+ self.rep = StringIO()
+ self.lastURL = None
+ self.lastEffectiveURL = None
+
+ # cookiejar defines the context
+ self.cj = self.context
+
+ self.c.setopt(pycurl.WRITEFUNCTION, self.write)
+ self.c.setopt(pycurl.HEADERFUNCTION, self.writeHeader)
+
+ # TODO: addAuth, addHeader
+
+ def initContext(self):
+ self.initHandle()
+
+ if self.config:
+ self.setInterface(self.config)
+ self.initOptions(self.config)
+
+ def initHandle(self):
+ """ sets common options to curl handle """
+
+ self.c.setopt(pycurl.FOLLOWLOCATION, 1)
+ self.c.setopt(pycurl.MAXREDIRS, 5)
+ self.c.setopt(pycurl.CONNECTTIMEOUT, 30)
+ self.c.setopt(pycurl.NOSIGNAL, 1)
+ self.c.setopt(pycurl.NOPROGRESS, 1)
+ if hasattr(pycurl, "AUTOREFERER"):
+ self.c.setopt(pycurl.AUTOREFERER, 1)
+ self.c.setopt(pycurl.SSL_VERIFYPEER, 0)
+ # Interval for low speed, detects connection loss, but can abort dl if hoster stalls the download
+ self.c.setopt(pycurl.LOW_SPEED_TIME, 45)
+ self.c.setopt(pycurl.LOW_SPEED_LIMIT, 5)
+
+ # don't save the cookies
+ self.c.setopt(pycurl.COOKIEFILE, "")
+ self.c.setopt(pycurl.COOKIEJAR, "")
+
+ #self.c.setopt(pycurl.VERBOSE, 1)
+
+ self.c.setopt(pycurl.USERAGENT,
+ "Mozilla/5.0 (Windows NT 6.1; Win64; x64;en; rv:5.0) Gecko/20110619 Firefox/5.0")
+ if pycurl.version_info()[7]:
+ self.c.setopt(pycurl.ENCODING, "gzip, deflate")
+ self.c.setopt(pycurl.HTTPHEADER, ["Accept: */*",
+ "Accept-Language: en-US,en",
+ "Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.7",
+ "Connection: keep-alive",
+ "Keep-Alive: 300",
+ "Expect:"])
+
+ def setInterface(self, options):
+
+ interface, proxy, ipv6 = options["interface"], options["proxies"], options["ipv6"]
+
+ if interface and interface.lower() != "none":
+ self.c.setopt(pycurl.INTERFACE, str(interface))
+
+ if proxy:
+ if proxy["type"] == "socks4":
+ self.c.setopt(pycurl.PROXYTYPE, pycurl.PROXYTYPE_SOCKS4)
+ elif proxy["type"] == "socks5":
+ self.c.setopt(pycurl.PROXYTYPE, pycurl.PROXYTYPE_SOCKS5)
+ else:
+ self.c.setopt(pycurl.PROXYTYPE, pycurl.PROXYTYPE_HTTP)
+
+ self.c.setopt(pycurl.PROXY, str(proxy["address"]))
+ self.c.setopt(pycurl.PROXYPORT, proxy["port"])
+
+ if proxy["username"]:
+ self.c.setopt(pycurl.PROXYUSERPWD, str("%s:%s" % (proxy["username"], proxy["password"])))
+
+ if ipv6:
+ self.c.setopt(pycurl.IPRESOLVE, pycurl.IPRESOLVE_WHATEVER)
+ else:
+ self.c.setopt(pycurl.IPRESOLVE, pycurl.IPRESOLVE_V4)
+
+ if "timeout" in options:
+ self.c.setopt(pycurl.LOW_SPEED_TIME, options["timeout"])
+
+ def initOptions(self, options):
+ """ Sets same options as available in pycurl """
+ for k, v in options.iteritems():
+ if hasattr(pycurl, k):
+ self.c.setopt(getattr(pycurl, k), v)
+
+ def setRequestContext(self, url, get, post, referer, cookies, multipart=False):
+ """ sets everything needed for the request """
+ url = myquote(url)
+
+ if get:
+ get = urlencode(get)
+ url = "%s?%s" % (url, get)
+
+ self.c.setopt(pycurl.URL, url)
+ self.lastURL = url
+
+ if post:
+ self.c.setopt(pycurl.POST, 1)
+ if not multipart:
+ if type(post) == unicode:
+ post = str(post) #unicode not allowed
+ elif type(post) == str:
+ pass
+ else:
+ post = myurlencode(post)
+
+ self.c.setopt(pycurl.POSTFIELDS, post)
+ else:
+ post = [(x, y.encode('utf8') if type(y) == unicode else y ) for x, y in post.iteritems()]
+ self.c.setopt(pycurl.HTTPPOST, post)
+ else:
+ self.c.setopt(pycurl.POST, 0)
+
+ if referer and self.lastURL:
+ self.c.setopt(pycurl.REFERER, str(self.lastURL))
+ else:
+ self.c.setopt(pycurl.REFERER, "")
+
+ if cookies:
+ self.c.setopt(pycurl.COOKIELIST, self.cj.output())
+ else:
+ # Magic string that erases all cookies
+ self.c.setopt(pycurl.COOKIELIST, "ALL")
+
+ # TODO: remove auth again
+ if "auth" in self.options:
+ self.c.setopt(pycurl.USERPWD, str(self.options["auth"]))
+
+
+ def load(self, url, get={}, post={}, referer=True, cookies=True, just_header=False, multipart=False, decode=False):
+ """ load and returns a given page """
+
+ self.setRequestContext(url, get, post, referer, cookies, multipart)
+
+ # TODO: use http/rfc message instead
+ self.header = ""
+
+ if "header" in self.options:
+ self.c.setopt(pycurl.HTTPHEADER, self.options["header"])
+
+ if just_header:
+ self.c.setopt(pycurl.FOLLOWLOCATION, 0)
+ self.c.setopt(pycurl.NOBODY, 1) #TODO: nobody= no post?
+
+ # overwrite HEAD request, we want a common request type
+ if post:
+ self.c.setopt(pycurl.CUSTOMREQUEST, "POST")
+ else:
+ self.c.setopt(pycurl.CUSTOMREQUEST, "GET")
+
+ try:
+ self.c.perform()
+ rep = self.header
+ finally:
+ self.c.setopt(pycurl.FOLLOWLOCATION, 1)
+ self.c.setopt(pycurl.NOBODY, 0)
+ self.c.unsetopt(pycurl.CUSTOMREQUEST)
+
+ else:
+ self.c.perform()
+ rep = self.getResponse()
+
+ self.c.setopt(pycurl.POSTFIELDS, "")
+ self.lastEffectiveURL = self.c.getinfo(pycurl.EFFECTIVE_URL)
+ self.code = self.verifyHeader()
+
+ if cookies:
+ self.parseCookies()
+
+ if decode:
+ rep = self.decodeResponse(rep)
+
+ return rep
+
+ def parseCookies(self):
+ for c in self.c.getinfo(pycurl.INFO_COOKIELIST):
+ #http://xiix.wordpress.com/2006/03/23/mozillafirefox-cookie-format
+ domain, flag, path, secure, expires, name, value = c.split("\t")
+ # http only was added in py 2.6
+ domain = domain.replace("#HttpOnly_", "")
+ self.cj.setCookie(domain, name, value, path, expires, secure)
+
+ def verifyHeader(self):
+ """ raise an exceptions on bad headers """
+ code = int(self.c.getinfo(pycurl.RESPONSE_CODE))
+ if code in bad_headers:
+ raise ResponseException(code, responses.get(code, "Unknown statuscode"))
+ return code
+
+ def getResponse(self):
+ """ retrieve response from string io """
+ if self.rep is None: return ""
+ value = self.rep.getvalue()
+ self.rep.close()
+ self.rep = StringIO()
+ return value
+
+ def decodeResponse(self, rep):
+ """ decode with correct encoding, relies on header """
+ header = self.header.splitlines()
+ encoding = "utf8" # default encoding
+
+ for line in header:
+ line = line.lower().replace(" ", "")
+ if not line.startswith("content-type:") or \
+ ("text" not in line and "application" not in line):
+ continue
+
+ none, delemiter, charset = line.rpartition("charset=")
+ if delemiter:
+ charset = charset.split(";")
+ if charset:
+ encoding = charset[0]
+
+ try:
+ #self.log.debug("Decoded %s" % encoding )
+ if lookup(encoding).name == 'utf-8' and rep.startswith(BOM_UTF8):
+ encoding = 'utf-8-sig'
+
+ decoder = getincrementaldecoder(encoding)("replace")
+ rep = decoder.decode(rep, True)
+
+ #TODO: html_unescape as default
+
+ except LookupError:
+ self.log.debug("No Decoder found for %s" % encoding)
+ except Exception:
+ self.log.debug("Error when decoding string from %s." % encoding)
+
+ return rep
+
+ def write(self, buf):
+ """ writes response """
+ if self.rep.tell() > 1000000 or self.doAbort:
+ rep = self.getResponse()
+ if self.doAbort: raise Abort()
+ f = open("response.dump", "wb")
+ f.write(rep)
+ f.close()
+ raise Exception("Loaded Url exceeded limit")
+
+ self.rep.write(buf)
+
+ def writeHeader(self, buf):
+ """ writes header """
+ self.header += buf
+
+ def reset(self):
+ self.cj.clear()
+ self.options.clear()
+
+ def close(self):
+ """ cleanup, unusable after this """
+ self.rep.close()
+ if hasattr(self, "cj"):
+ del self.cj
+ if hasattr(self, "c"):
+ self.c.close()
+ del self.c \ No newline at end of file
diff --git a/pyload/plugins/network/DefaultRequest.py b/pyload/plugins/network/DefaultRequest.py
new file mode 100644
index 000000000..dce486ea5
--- /dev/null
+++ b/pyload/plugins/network/DefaultRequest.py
@@ -0,0 +1,9 @@
+# -*- coding: utf-8 -*-
+
+from CurlRequest import CurlRequest
+from CurlDownload import CurlDownload
+
+__version__ = "0.1"
+
+DefaultRequest = CurlRequest
+DefaultDownload = CurlDownload \ No newline at end of file