2007-08-31 16:06:31 +04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2008-07-04 11:48:15 +04:00
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# * Neither the name of the author nor the names of its contributors may be
|
|
|
|
# used to endorse or promote products derived from this software without
|
|
|
|
# specific prior written permission.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
2008-07-03 19:09:39 +04:00
|
|
|
|
2012-09-15 17:54:09 +04:00
|
|
|
#VERSION: 1.31
|
2007-08-31 16:06:31 +04:00
|
|
|
|
|
|
|
# Author:
|
|
|
|
# Fabien Devaux <fab AT gnux DOT info>
|
|
|
|
# Contributors:
|
|
|
|
# Christophe Dumez <chris@qbittorrent.org> (qbittorrent integration)
|
|
|
|
# Thanks to gab #gcu @ irc.freenode.net (multipage support on PirateBay)
|
|
|
|
# Thanks to Elias <gekko04@users.sourceforge.net> (torrentreactor and isohunt search engines)
|
|
|
|
#
|
|
|
|
# Licence: BSD
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import threading
|
|
|
|
import os
|
|
|
|
import glob
|
|
|
|
|
2012-09-15 17:54:09 +04:00
|
|
|
import fix_encoding
|
|
|
|
|
2007-08-31 16:06:31 +04:00
|
|
|
THREADED = True
|
2009-08-25 06:31:36 +04:00
|
|
|
CATEGORIES = ('all', 'movies', 'tv', 'music', 'games', 'anime', 'software', 'pictures', 'books')
|
2007-08-31 16:06:31 +04:00
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Every engine should have a "search" method taking
|
|
|
|
# a space-free string as parameter (ex. "family+guy")
|
|
|
|
# it should call prettyPrinter() with a dict as parameter.
|
|
|
|
# The keys in the dict must be: link,name,size,seeds,leech,engine_url
|
|
|
|
# As a convention, try to list results by decrasing number of seeds or similar
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
supported_engines = []
|
|
|
|
|
|
|
|
engines = glob.glob(os.path.join(os.path.dirname(__file__), 'engines','*.py'))
|
|
|
|
for engine in engines:
|
|
|
|
e = engine.split(os.sep)[-1][:-3]
|
|
|
|
if len(e.strip()) == 0: continue
|
|
|
|
if e.startswith('_'): continue
|
|
|
|
try:
|
|
|
|
exec "from engines.%s import %s"%(e,e)
|
|
|
|
supported_engines.append(e)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2009-08-25 06:31:36 +04:00
|
|
|
def engineToXml(short_name):
|
|
|
|
xml = "<%s>\n"%short_name
|
|
|
|
exec "engine = %s()"%short_name
|
|
|
|
xml += "<name>%s</name>\n"%engine.name
|
|
|
|
xml += "<url>%s</url>\n"%engine.url
|
|
|
|
xml += "<categories>"
|
|
|
|
if hasattr(engine, 'supported_categories'):
|
|
|
|
supported_categories = engine.supported_categories.keys()
|
|
|
|
supported_categories.remove('all')
|
|
|
|
xml += " ".join(supported_categories)
|
|
|
|
xml += "</categories>\n"
|
|
|
|
xml += "</%s>\n"%short_name
|
|
|
|
return xml
|
|
|
|
|
|
|
|
def displayCapabilities():
|
|
|
|
"""
|
|
|
|
Display capabilities in XML format
|
|
|
|
<capabilities>
|
|
|
|
<engine_short_name>
|
|
|
|
<name>long name</name>
|
|
|
|
<url>http://example.com</url>
|
|
|
|
<categories>movies music games</categories>
|
|
|
|
</engine_short_name>
|
|
|
|
</capabilities>
|
|
|
|
"""
|
|
|
|
xml = "<capabilities>"
|
|
|
|
for short_name in supported_engines:
|
|
|
|
xml += engineToXml(short_name)
|
|
|
|
xml += "</capabilities>"
|
|
|
|
print xml
|
|
|
|
|
2007-08-31 16:06:31 +04:00
|
|
|
class EngineLauncher(threading.Thread):
|
2009-08-25 06:31:36 +04:00
|
|
|
def __init__(self, engine, what, cat='all'):
|
2007-08-31 16:06:31 +04:00
|
|
|
threading.Thread.__init__(self)
|
|
|
|
self.engine = engine
|
|
|
|
self.what = what
|
2009-08-25 06:31:36 +04:00
|
|
|
self.cat = cat
|
2007-08-31 16:06:31 +04:00
|
|
|
def run(self):
|
2009-08-25 06:31:36 +04:00
|
|
|
if hasattr(self.engine, 'supported_categories'):
|
|
|
|
if self.cat == 'all' or self.cat in self.engine.supported_categories.keys():
|
|
|
|
self.engine.search(self.what, self.cat)
|
|
|
|
elif self.cat == 'all':
|
|
|
|
self.engine.search(self.what)
|
2007-08-31 16:06:31 +04:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2012-09-15 17:54:09 +04:00
|
|
|
# Make sure we enforce utf-8 encoding
|
|
|
|
fix_encoding.fix_encoding()
|
|
|
|
|
2007-08-31 16:06:31 +04:00
|
|
|
if len(sys.argv) < 2:
|
2009-08-25 06:31:36 +04:00
|
|
|
raise SystemExit('./nova2.py [all|engine1[,engine2]*] <category> <keywords>\navailable engines: %s'%
|
2007-08-31 16:06:31 +04:00
|
|
|
(','.join(supported_engines)))
|
|
|
|
|
|
|
|
if len(sys.argv) == 2:
|
2009-08-25 06:31:36 +04:00
|
|
|
if sys.argv[1] == "--capabilities":
|
|
|
|
displayCapabilities()
|
2007-08-31 16:06:31 +04:00
|
|
|
sys.exit(0)
|
|
|
|
else:
|
2009-08-25 06:31:36 +04:00
|
|
|
raise SystemExit('./nova.py [all|engine1[,engine2]*] <category> <keywords>\navailable engines: %s'%
|
2007-08-31 16:06:31 +04:00
|
|
|
(','.join(supported_engines)))
|
|
|
|
|
|
|
|
engines_list = [e.lower() for e in sys.argv[1].strip().split(',')]
|
|
|
|
|
|
|
|
if 'all' in engines_list:
|
|
|
|
engines_list = supported_engines
|
2009-08-25 06:31:36 +04:00
|
|
|
|
|
|
|
cat = sys.argv[2].lower()
|
|
|
|
|
|
|
|
if cat not in CATEGORIES:
|
|
|
|
raise SystemExit('Invalid category!')
|
|
|
|
|
|
|
|
what = '+'.join(sys.argv[3:])
|
|
|
|
|
2007-08-31 16:06:31 +04:00
|
|
|
threads = []
|
|
|
|
for engine in engines_list:
|
2009-03-27 17:34:30 +03:00
|
|
|
try:
|
2007-08-31 16:06:31 +04:00
|
|
|
if THREADED:
|
2009-08-25 06:31:36 +04:00
|
|
|
exec "l = EngineLauncher(%s(), what, cat)"%engine
|
2007-08-31 16:06:31 +04:00
|
|
|
threads.append(l)
|
|
|
|
l.start()
|
|
|
|
else:
|
2009-08-25 06:31:36 +04:00
|
|
|
exec "e = %s()"%engine
|
|
|
|
if hasattr(engine, 'supported_categories'):
|
|
|
|
if cat == 'all' or cat in e.supported_categories.keys():
|
|
|
|
e.search(what, cat)
|
|
|
|
elif self.cat == 'all':
|
|
|
|
e.search(what)
|
|
|
|
engine().search(what, cat)
|
2009-03-27 17:34:30 +03:00
|
|
|
except:
|
|
|
|
pass
|
2007-08-31 16:06:31 +04:00
|
|
|
if THREADED:
|
|
|
|
for t in threads:
|
|
|
|
t.join()
|