# -*- coding: utf-8 -*- from __future__ import with_statement import StringIO import pycurl import time import urllib try: from PIL import Image except ImportError: import Image from module.network.RequestFactory import getRequest as get_request from module.plugins.internal.Addon import Addon, threaded class CaptchaBrotherhoodException(Exception): def __init__(self, err): self.err = err def get_code(self): return self.err def __str__(self): return "" % self.err def __repr__(self): return "" % self.err class CaptchaBrotherhood(Addon): __name__ = "CaptchaBrotherhood" __type__ = "hook" __version__ = "0.11" __status__ = "testing" __config__ = [("activated" , "bool" , "Activated" , False), ("username" , "str" , "Username" , "" ), ("password" , "password", "Password" , "" ), ("check_client", "bool" , "Don't use if client is connected", True )] __description__ = """Send captchas to CaptchaBrotherhood.com""" __license__ = "GPLv3" __authors__ = [("RaNaN" , "RaNaN@pyload.org" ), ("zoidberg", "zoidberg@mujmail.cz")] API_URL = "http://www.captchabrotherhood.com/" def get_credits(self): res = self.load(self.API_URL + "askCredits.aspx", get={'username': self.get_config('username'), 'password': self.get_config('password')}) if not res.startswith("OK"): raise CaptchaBrotherhoodException(res) else: credits = int(res[3:]) self.log_info(_("%d credits left") % credits) self.info['credits'] = credits return credits def submit(self, captcha, captchaType="file", match=None): try: img = Image.open(captcha) output = StringIO.StringIO() self.log_debug("CAPTCHA IMAGE", img, img.format, img.mode) if img.format in ("GIF", "JPEG"): img.save(output, img.format) else: if img.mode != "RGB": img = img.convert("RGB") img.save(output, "JPEG") data = output.getvalue() output.close() except Exception, e: raise CaptchaBrotherhoodException("Reading or converting captcha image failed: %s" % e) req = get_request() url = "%ssendNewCaptcha.aspx?%s" % (self.API_URL, urllib.urlencode({'username' : self.get_config('username'), 'password' : self.get_config('password'), 'captchaSource': "pyLoad", 'timeout' : "80"})) req.c.setopt(pycurl.URL, url) req.c.setopt(pycurl.POST, 1) req.c.setopt(pycurl.POSTFIELDS, data) req.c.setopt(pycurl.HTTPHEADER, ["Content-Type: text/html"]) try: req.c.perform() res = req.getResponse() except Exception, e: raise CaptchaBrotherhoodException("Submit captcha image failed") req.close() if not res.startswith("OK"): raise CaptchaBrotherhoodException(res[1]) ticket = res[3:] for _i in xrange(15): time.sleep(5) res = self.api_response("askCaptchaResult", ticket) if res.startswith("OK-answered"): return ticket, res[12:] raise CaptchaBrotherhoodException("No solution received in time") def api_response(self, api, ticket): res = self.load("%s%s.aspx" % (self.API_URL, api), get={'username': self.get_config('username'), 'password': self.get_config('password'), 'captchaID': ticket}) if not res.startswith("OK"): raise CaptchaBrotherhoodException("Unknown response: %s" % res) return res def captcha_task(self, task): if "service" in task.data: return False if not task.isTextual(): return False if not self.get_config('username') or not self.get_config('password'): return False if self.pyload.isClientConnected() and self.get_config('check_client'): return False if self.get_credits() > 10: task.handler.append(self) task.data['service'] = self.classname task.setWaiting(100) self._process_captcha(task) else: self.log_info(_("Your CaptchaBrotherhood Account has not enough credits")) def captcha_invalid(self, task): if task.data['service'] is self.classname and "ticket" in task.data: res = self.api_response("complainCaptcha", task.data['ticket']) @threaded def _process_captcha(self, task): c = task.captchaFile try: ticket, result = self.submit(c) except CaptchaBrotherhoodException, e: task.error = e.get_code() return task.data['ticket'] = ticket task.setResult(result)