mirror of https://github.com/evilhero/mylar
180 lines
6.3 KiB
Python
Executable File
180 lines
6.3 KiB
Python
Executable File
import os
|
|
import re
|
|
from urlparse import urlparse
|
|
|
|
from lib.rtorrent import RTorrent
|
|
|
|
import mylar
|
|
from mylar import logger, helpers
|
|
|
|
class TorrentClient(object):
|
|
def __init__(self):
|
|
self.conn = None
|
|
|
|
def getVerifySsl(self):
|
|
# Ensure verification has been enabled
|
|
if not mylar.RTORRENT_VERIFY:
|
|
return False
|
|
|
|
# Use ca bundle if defined
|
|
if mylar.RTORRENT_CA_BUNDLE and os.path.exists(ca_bundle):
|
|
return mylar.RTORRENT_CA_BUNDLE
|
|
|
|
# Use default ssl verification
|
|
return True
|
|
|
|
def connect(self, host, username, password, auth):
|
|
if self.conn is not None:
|
|
return self.conn
|
|
|
|
if not host:
|
|
return False
|
|
|
|
url = helpers.cleanHost(host, protocol = True, ssl = mylar.RTORRENT_SSL)
|
|
|
|
# Automatically add '+https' to 'httprpc' protocol if SSL is enabled
|
|
if mylar.RTORRENT_SSL and url.startswith('httprpc://'):
|
|
url = url.replace('httprpc://', 'httprpc+https://')
|
|
|
|
parsed = urlparse(url)
|
|
|
|
# rpc_url is only used on http/https scgi pass-through
|
|
if parsed.scheme in ['http', 'https']:
|
|
url += mylar.RTORRENT_RPC_URL
|
|
|
|
logger.info(url)
|
|
|
|
if username and password:
|
|
try:
|
|
self.conn = RTorrent(
|
|
url,(auth, username, password),
|
|
verify_server=True,
|
|
verify_ssl=self.getVerifySsl()
|
|
)
|
|
except:
|
|
return False
|
|
else:
|
|
try:
|
|
self.conn = RTorrent(host)
|
|
except:
|
|
return False
|
|
|
|
return self.conn
|
|
|
|
def find_torrent(self, hash):
|
|
return self.conn.find_torrent(hash)
|
|
|
|
def get_torrent (self, torrent):
|
|
torrent_files = []
|
|
torrent_directory = os.path.normpath(torrent.directory)
|
|
try:
|
|
for f in torrent.get_files():
|
|
if not os.path.normpath(f.path).startswith(torrent_directory):
|
|
file_path = os.path.join(torrent_directory, f.path.lstrip('/'))
|
|
else:
|
|
file_path = f.path
|
|
|
|
torrent_files.append(file_path)
|
|
|
|
torrent_info = {
|
|
'hash': torrent.info_hash,
|
|
'name': torrent.name,
|
|
'label': torrent.get_custom1() if torrent.get_custom1() else '',
|
|
'folder': torrent_directory,
|
|
'completed': torrent.complete,
|
|
'files': torrent_files,
|
|
'upload_total': torrent.get_up_total(),
|
|
'download_total': torrent.get_down_total(),
|
|
'ratio': torrent.get_ratio(),
|
|
'total_filesize': torrent.get_size_bytes(),
|
|
'time_started': torrent.get_time_started()
|
|
}
|
|
|
|
except Exception:
|
|
raise
|
|
|
|
return torrent_info if torrent_info else False
|
|
|
|
def load_torrent(self, filepath):
|
|
start = bool(mylar.RTORRENT_STARTONLOAD)
|
|
|
|
if filepath.startswith('magnet'):
|
|
logger.info('torrent magnet link set to : ' + filepath)
|
|
torrent_hash = re.findall('urn:btih:([\w]{32,40})', filepath)[0].upper()
|
|
# Send request to rTorrent
|
|
try:
|
|
#cannot verify_load magnet as it might take a very very long time for it to retrieve metadata
|
|
torrent = self.conn.load_magnet(filepath, torrent_hash, verify_load=True)
|
|
if not torrent:
|
|
logger.error('Unable to find the torrent, did it fail to load?')
|
|
return False
|
|
except Exception as err:
|
|
logger.error('Failed to send magnet to rTorrent: %s', err)
|
|
return False
|
|
else:
|
|
logger.info('Torrent successfully loaded into rtorrent using magnet link as source.')
|
|
else:
|
|
logger.info('filepath to torrent file set to : ' + filepath)
|
|
try:
|
|
torrent = self.conn.load_torrent(filepath, verify_load=True)
|
|
if not torrent:
|
|
logger.error('Unable to find the torrent, did it fail to load?')
|
|
return False
|
|
except Exception as err:
|
|
logger.error('Failed to send torrent to rTorrent: %s', err)
|
|
return False
|
|
|
|
#we can cherrypick the torrents here if required and if it's a pack (0day instance)
|
|
#torrent.get_files() will return list of files in torrent
|
|
#f.set_priority(0,1,2)
|
|
#for f in torrent.get_files():
|
|
# logger.info('torrent_get_files: %s' % f)
|
|
# f.set_priority(0) #set them to not download just to see if this works...
|
|
#torrent.updated_priorities()
|
|
|
|
if mylar.RTORRENT_LABEL:
|
|
torrent.set_custom(1, mylar.RTORRENT_LABEL)
|
|
logger.info('Setting label for torrent to : ' + mylar.RTORRENT_LABEL)
|
|
|
|
if mylar.RTORRENT_DIRECTORY:
|
|
torrent.set_directory(mylar.RTORRENT_DIRECTORY)
|
|
logger.info('Setting directory for torrent to : ' + mylar.RTORRENT_DIRECTORY)
|
|
|
|
logger.info('Successfully loaded torrent.')
|
|
|
|
#note that if set_directory is enabled, the torrent has to be started AFTER it's loaded or else it will give chunk errors and not seed
|
|
if start:
|
|
logger.info('[' + str(start) + '] Now starting torrent.')
|
|
torrent.start()
|
|
else:
|
|
logger.info('[' + str(start) + '] Not starting torrent due to configuration setting.')
|
|
return True
|
|
|
|
def start_torrent(self, torrent):
|
|
return torrent.start()
|
|
|
|
def stop_torrent(self, torrent):
|
|
return torrent.stop()
|
|
|
|
def delete_torrent(self, torrent):
|
|
deleted = []
|
|
try:
|
|
for file_item in torrent.get_files():
|
|
file_path = os.path.join(torrent.directory, file_item.path)
|
|
os.unlink(file_path)
|
|
deleted.append(file_item.path)
|
|
|
|
if torrent.is_multi_file() and torrent.directory.endswith(torrent.name):
|
|
try:
|
|
for path, _, _ in os.walk(torrent.directory, topdown=False):
|
|
os.rmdir(path)
|
|
deleted.append(path)
|
|
except:
|
|
pass
|
|
except Exception:
|
|
raise
|
|
|
|
torrent.erase()
|
|
|
|
return deleted
|