summaryrefslogtreecommitdiffstats
path: root/module/plugins/crypter/LinkSaveIn.py
diff options
context:
space:
mode:
Diffstat (limited to 'module/plugins/crypter/LinkSaveIn.py')
-rw-r--r--module/plugins/crypter/LinkSaveIn.py167
1 files changed, 133 insertions, 34 deletions
diff --git a/module/plugins/crypter/LinkSaveIn.py b/module/plugins/crypter/LinkSaveIn.py
index 8809bd7d1..33d49608c 100644
--- a/module/plugins/crypter/LinkSaveIn.py
+++ b/module/plugins/crypter/LinkSaveIn.py
@@ -2,6 +2,7 @@
from Crypto.Cipher import AES
from module.plugins.Crypter import Crypter
+from module.unescape import unescape
import base64
import binascii
import re
@@ -9,8 +10,8 @@ import re
class LinkSaveIn(Crypter):
__name__ = "LinkSaveIn"
__type__ = "crypter"
- __pattern__ = r"http://(www\.)?linksave.in/([a-z0-9]+)$"
- __version__ = "1.01"
+ __pattern__ = r"http://(www\.)?linksave.in/(?P<id>\w+)$"
+ __version__ = "2.0"
__description__ = """LinkSave.in Crypter Plugin"""
__author_name__ = ("fragonib")
__author_mail__ = ("fragonib[AT]yahoo[DOT]es")
@@ -18,64 +19,162 @@ class LinkSaveIn(Crypter):
# Constants
_JK_KEY_ = "jk"
_CRYPTED_KEY_ = "crypted"
+ HOSTER_DOMAIN = "linksave.in"
+
+ def setup(self):
+ self.html = None
+ self.fileid = None
+ self.captcha = False
+ self.package = None
def decrypt(self, pyfile):
- # Request page
- self.html = self.load(pyfile.url)
- if not self.fileExists():
+ # Init
+ self.package = pyfile.package()
+ self.fileid = re.match(self.__pattern__, pyfile.url).group('id')
+ self.req.cj.setCookie(self.HOSTER_DOMAIN, "Linksave_Language", "english")
+
+ # Request package
+ self.html = self.load(self.pyfile.url, cookies=True)
+ if not self.isOnline():
self.offline()
-
- # Handle captcha protection
- self.handleCaptcha()
+
+ # Check for protection
+ if self.isPasswordProtected():
+ self.unlockPasswordProtection()
+ self.handleErrors()
+
+ if self.isCaptchaProtected():
+ self.captcha = True
+ self.unlockCaptchaProtection()
+ self.handleErrors()
# Get package name and folder
- (package_name, folder_name) = self.getPackageNameAndFolder()
+ (package_name, folder_name) = self.getPackageInfo()
- # Get package links
- (crypted, jk) = self.getCipherParams()
- package_links = self.getLinks(crypted, jk)
+ # Extract package links
+ package_links = []
+ package_links.extend(self.handleWebLinks())
+ package_links.extend(self.handleContainers())
+ package_links.extend(self.handleCNL2())
+ package_links = set(package_links)
# Pack
self.packages = [(package_name, package_links, folder_name)]
- def fileExists(self):
- if "<title>LinkSave.in - Error 404</title>" in self.html:
+ def isOnline(self):
+ if "<big>Error 404 - Folder not found!</big>" in self.html:
self.log.debug("%s: File not found" % self.__name__)
return False
return True
- def getPackageNameAndFolder(self):
+ def isPasswordProtected(self):
+ if re.search(r'''<input.*?type="password"''', self.html):
+ self.log.debug("%s: Links are password protected" % self.__name__)
+ return True
+
+ def isCaptchaProtected(self):
+ if "<b>Captcha:</b>" in self.html:
+ self.log.debug("%s: Links are captcha protected" % self.__name__)
+ return True
+ return False
+
+ def unlockPasswordProtection(self):
+ password = self.package.password
+ self.log.debug("%s: Submitting password [%s] for protected links" % (self.__name__, password))
+ post = {"id": self.fileid, "besucherpasswort": self.package.password, 'login': 'submit'}
+ self.html = self.load(self.pyfile.url, post=post)
+
+ def unlockCaptchaProtection(self):
+ hash = re.search(r'name="hash" value="([^"]+)', self.html).group(1)
+ captchaUrl = re.search(r'src=".(/captcha/cap.php\?hsh=[^"]+)', self.html).group(1)
+ code = self.decryptCaptcha("http://linksave.in" + captchaUrl, forceUser=True)
+ self.html = self.load(self.pyfile.url, post={"id": self.fileid, "hash": hash, "code": code})
+
+ def getPackageInfo(self):
name = self.pyfile.package().name
folder = self.pyfile.package().folder
- self.log.debug("%s: Default to pyfile name [%s] and folder [%s] for package" % (self.__name__, name, folder))
+ self.log.debug("%s: Defaulting to pyfile name [%s] and folder [%s] for package" % (self.__name__, name, folder))
return name, folder
-
- def handleCaptcha(self):
- if "<b>Captcha:</b>" in self.html:
- id = re.search(r'name="id" value="([^"]+)', self.html).group(1)
- hash = re.search(r'name="hash" value="([^"]+)', self.html).group(1)
- url = re.search(r'src=".(/captcha/cap.php\?hsh=[^"]+)', self.html).group(1)
- value = self.decryptCaptcha("http://linksave.in" + url, forceUser=True)
- self.html = self.load(self.pyfile.url, post={"id": id, "hash": hash, "code": value})
- def getCipherParams(self):
+ def handleErrors(self):
+ if "The visitorpassword you have entered is wrong" in self.html:
+ self.log.debug("%s: Incorrect password, please set right password on 'Edit package' form and retry" % self.__name__)
+ self.fail("Incorrect password, please set right password on 'Edit package' form and retry")
+
+ if self.captcha:
+ if "Wrong code. Please retry" in self.html:
+ self.log.debug("%s: Invalid captcha, retrying" % self.__name__)
+ self.invalidCaptcha()
+ self.retry()
+ else:
+ self.correctCaptcha()
+
+ def handleWebLinks(self):
+ package_links = []
+ self.log.debug("%s: Handling Web links" % self.__name__)
+
+ #@TODO: Gather paginated web links
+ pattern = r'<a href="http://linksave\.in/(\w{43})"'
+ ids = re.findall(pattern, self.html)
+ self.log.debug("%s: Decrypting %d Web links" % (self.__name__, len(ids)))
+ for i, id in enumerate(ids):
+ try:
+ webLink = "http://linksave.in/%s" % id
+ self.log.debug("%s: Decrypting Web link %d, %s" % (self.__name__, i+1, webLink))
+ fwLink = "http://linksave.in/fw-%s" % id
+ response = self.load(fwLink)
+ jscode = re.findall(r'<script type="text/javascript">(.*)</script>', response)[-1]
+ jseval = self.js.eval("document = { write: function(e) { return e; } }; %s" % jscode)
+ dlLink = re.search(r'http://linksave\.in/dl-\w+', jseval).group(0)
+ self.log.debug("%s: JsEngine returns value [%s] for redirection link" % (self.__name__, dlLink))
+ response = self.load(dlLink)
+ link = unescape(re.search(r'<iframe src="(.+?)"', response).group(1))
+ package_links.append(link)
+ except Exception, detail:
+ self.log.debug("%s: Error decrypting Web link %s, %s" % (self.__name__, webLink, detail))
+ return package_links
+
+ def handleContainers(self):
+ package_links = []
+ self.log.debug("%s: Handling Container links" % self.__name__)
+
+ pattern = r"\('(?:rsdf|ccf|dlc)_link'\).href=unescape\('(.*?\.(?:rsdf|ccf|dlc))'\)"
+ containersLinks = re.findall(pattern, self.html)
+ self.log.debug("%s: Decrypting %d Container links" % (self.__name__, len(containersLinks)))
+ for containerLink in containersLinks:
+ link = "http://linksave.in/%s" % unescape(containerLink)
+ package_links.append(link)
+ return package_links
+
+ def handleCNL2(self):
+ package_links = []
+ self.log.debug("%s: Handling CNL2 links" % self.__name__)
+
+ if 'cnl2_load' in self.html:
+ try:
+ (vcrypted, vjk) = self._getCipherParams()
+ for (crypted, jk) in zip(vcrypted, vjk):
+ package_links.extend(self._getLinks(crypted, jk))
+ except:
+ self.fail("Unable to decrypt CNL2 links")
+ return package_links
+
+ def _getCipherParams(self):
# Get jk
- jk_re = r'<INPUT(.*?)NAME="%s"(.*?)VALUE="(?P<jk>.*?)"' % LinkSaveIn._JK_KEY_
- m = re.search(jk_re, self.html)
- jk = m.group('jk')
+ jk_re = r'<INPUT.*?NAME="%s".*?VALUE="(.*?)"' % LinkSaveIn._JK_KEY_
+ vjk = re.findall(jk_re, self.html)
# Get crypted
- crypted_re = r'<INPUT(.*?)NAME="%s"(.*?)VALUE="(?P<crypted>.*?)"' % LinkSaveIn._CRYPTED_KEY_
- m = re.search(crypted_re, self.html)
- crypted = m.group('crypted')
+ crypted_re = r'<INPUT.*?NAME="%s".*?VALUE="(.*?)"' % LinkSaveIn._CRYPTED_KEY_
+ vcrypted = re.findall(crypted_re, self.html)
# Log and return
- self.log.debug("%s: Javascript cipher key function [%s]" % (self.__name__, jk))
- return crypted, jk
+ self.log.debug("%s: Detected %d crypted blocks" % (self.__name__, len(vcrypted)))
+ return vcrypted, vjk
- def getLinks(self, crypted, jk):
+ def _getLinks(self, crypted, jk):
# Get key
jreturn = self.js.eval("%s f()" % jk)