#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
#Copyright (C) 2012 RaNaN
#
#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 .
#
###
from __future__ import with_statement
from getopt import GetoptError, getopt
import pyload.common.pylgettext as gettext
import os
from os import _exit
from os.path import join, exists, abspath, basename
import sys
from sys import exit
from threading import Thread, Lock
from time import sleep
from traceback import print_exc
import ConfigParser
from codecs import getwriter
if os.name == "nt":
enc = "cp850"
else:
enc = "utf8"
sys.stdout = getwriter(enc)(sys.stdout, errors="replace")
from pyload import InitHomeDir
from pyload.cli.printer import *
from pyload.cli import AddPackage, ManageFiles
from pyload.Api import Destination
from pyload.utils import formatSize, decode
from pyload.remote.thriftbackend.ThriftClient import ThriftClient, NoConnection, NoSSL, WrongLogin, ConnectionClosed
from pyload.lib.Getch import Getch
from pyload.lib.rename_process import renameProcess
class Cli:
def __init__(self, client, command):
self.client = client
self.command = command
if not self.command:
renameProcess('pyLoadCli')
self.getch = Getch()
self.input = ""
self.inputline = 0
self.lastLowestLine = 0
self.menuline = 0
self.lock = Lock()
#processor funcions, these will be changed dynamically depending on control flow
self.headerHandler = self #the download status
self.bodyHandler = self #the menu section
self.inputHandler = self
os.system("clear")
println(1, blue("py") + yellow("Load") + white(_(" Command Line Interface")))
println(2, "")
self.thread = RefreshThread(self)
self.thread.start()
self.start()
else:
self.processCommand()
def reset(self):
""" reset to initial main menu """
self.input = ""
self.headerHandler = self.bodyHandler = self.inputHandler = self
def start(self):
""" main loop. handle input """
while True:
#inp = raw_input()
inp = self.getch.impl()
if ord(inp) == 3:
os.system("clear")
sys.exit() # ctrl + c
elif ord(inp) == 13: #enter
try:
self.lock.acquire()
self.inputHandler.onEnter(self.input)
except Exception, e:
println(2, red(e))
finally:
self.lock.release()
elif ord(inp) == 127:
self.input = self.input[:-1] #backspace
try:
self.lock.acquire()
self.inputHandler.onBackSpace()
finally:
self.lock.release()
elif ord(inp) == 27: #ugly symbol
pass
else:
self.input += inp
try:
self.lock.acquire()
self.inputHandler.onChar(inp)
finally:
self.lock.release()
self.inputline = self.bodyHandler.renderBody(self.menuline)
self.renderFooter(self.inputline)
def refresh(self):
"""refresh screen"""
println(1, blue("py") + yellow("Load") + white(_(" Command Line Interface")))
println(2, "")
self.lock.acquire()
self.menuline = self.headerHandler.renderHeader(3) + 1
println(self.menuline - 1, "")
self.inputline = self.bodyHandler.renderBody(self.menuline)
self.renderFooter(self.inputline)
self.lock.release()
def setInput(self, string=""):
self.input = string
def setHandler(self, klass):
#create new handler with reference to cli
self.bodyHandler = self.inputHandler = klass(self)
self.input = ""
def renderHeader(self, line):
""" prints download status """
#print updated information
# print "\033[J" #clear screen
# self.println(1, blue("py") + yellow("Load") + white(_(" Command Line Interface")))
# self.println(2, "")
# self.println(3, white(_("%s Downloads:") % (len(data))))
data = self.client.statusDownloads()
speed = 0
println(line, white(_("%s Downloads:") % (len(data))))
line += 1
for download in data:
if download.status == 12: # downloading
percent = download.percent
z = percent / 4
speed += download.speed
println(line, cyan(download.name))
line += 1
println(line,
blue("[") + yellow(z * "#" + (25 - z) * " ") + blue("] ") + green(str(percent) + "%") + _(
" Speed: ") + green(formatSize(download.speed) + "/s") + _(" Size: ") + green(
download.format_size) + _(" Finished in: ") + green(download.format_eta) + _(
" ID: ") + green(download.fid))
line += 1
if download.status == 5:
println(line, cyan(download.name))
line += 1
println(line, _("waiting: ") + green(download.format_wait))
line += 1
println(line, "")
line += 1
status = self.client.statusServer()
if status.pause:
paused = _("Status:") + " " + red(_("paused"))
else:
paused = _("Status:") + " " + red(_("running"))
println(line,"%s %s: %s %s: %s %s: %s" % (
paused, _("total Speed"), red(formatSize(speed) + "/s"), _("Files in queue"), red(
status.queue), _("Total"), red(status.total)))
return line + 1
def renderBody(self, line):
""" prints initial menu """
println(line, white(_("Menu:")))
println(line + 1, "")
println(line + 2, mag("1.") + _(" Add Links"))
println(line + 3, mag("2.") + _(" Manage Queue"))
println(line + 4, mag("3.") + _(" Manage Collector"))
println(line + 5, mag("4.") + _(" (Un)Pause Server"))
println(line + 6, mag("5.") + _(" Kill Server"))
println(line + 7, mag("6.") + _(" Quit"))
return line + 8
def renderFooter(self, line):
""" prints out the input line with input """
println(line, "")
line += 1
println(line, white(" Input: ") + decode(self.input))
#clear old output
if line < self.lastLowestLine:
for i in range(line + 1, self.lastLowestLine + 1):
println(i, "")
self.lastLowestLine = line
#set cursor to position
print "\033[" + str(self.inputline) + ";0H"
def onChar(self, char):
""" default no special handling for single chars """
if char == "1":
self.setHandler(AddPackage)
elif char == "2":
self.setHandler(ManageFiles)
elif char == "3":
self.setHandler(ManageFiles)
self.bodyHandler.target = Destination.Collector
elif char == "4":
self.client.togglePause()
self.setInput()
elif char == "5":
self.client.kill()
self.client.close()
sys.exit()
elif char == "6":
os.system('clear')
sys.exit()
def onEnter(self, inp):
pass
def onBackSpace(self):
pass
def processCommand(self):
command = self.command[0]
args = []
if len(self.command) > 1:
args = self.command[1:]
if command == "status":
files = self.client.statusDownloads()
if not files:
print "No downloads running."
for download in files:
if download.status == 12: # downloading
print print_status(download)
print "\tDownloading: %s @ %s/s\t %s (%s%%)" % (
download.format_eta, formatSize(download.speed), formatSize(download.size - download.bleft),
download.percent)
elif download.status == 5:
print print_status(download)
print "\tWaiting: %s" % download.format_wait
else:
print print_status(download)
elif command == "queue":
print_packages(self.client.getQueueData())
elif command == "collector":
print_packages(self.client.getCollectorData())
elif command == "add":
if len(args) < 2:
print _("Please use this syntax: add ...")
return
self.client.addPackage(args[0], args[1:], Destination.Queue, "")
elif command == "add_coll":
if len(args) < 2:
print _("Please use this syntax: add ...")
return
self.client.addPackage(args[0], args[1:], Destination.Collector, "")
elif command == "del_file":
self.client.deleteFiles([int(x) for x in args])
print "Files deleted."
elif command == "del_package":
self.client.deletePackages([int(x) for x in args])
print "Packages deleted."
elif command == "move":
for pid in args:
pack = self.client.getPackageInfo(int(pid))
self.client.movePackage((pack.dest + 1) % 2, pack.pid)
elif command == "check":
print _("Checking %d links:") % len(args)
print
rid = self.client.checkOnlineStatus(args).rid
self.printOnlineCheck(self.client, rid)
elif command == "check_container":
path = args[0]
if not exists(join(owd, path)):
print _("File does not exists.")
return
f = open(join(owd, path), "rb")
content = f.read()
f.close()
rid = self.client.checkOnlineStatusContainer([], basename(f.name), content).rid
self.printOnlineCheck(self.client, rid)
elif command == "pause":
self.client.pauseServer()
elif command == "unpause":
self.client.unpauseServer()
elif command == "toggle":
self.client.togglePause()
elif command == "kill":
self.client.kill()
elif command == "restart_file":
for x in args:
self.client.restartFile(int(x))
print "Files restarted."
elif command == "restart_package":
for pid in args:
self.client.restartPackage(int(pid))
print "Packages restarted."
else:
print_commands()
def printOnlineCheck(self, client, rid):
while True:
sleep(1)
result = client.pollResults(rid)
for url, status in result.data.iteritems():
if status.status == 2: check = "Online"
elif status.status == 1: check = "Offline"
else: check = "Unknown"
print "%-45s %-12s\t %-15s\t %s" % (status.name, formatSize(status.size), status.plugin, check)
if result.rid == -1: break
class RefreshThread(Thread):
def __init__(self, cli):
Thread.__init__(self)
self.setDaemon(True)
self.cli = cli
def run(self):
while True:
sleep(1)
try:
self.cli.refresh()
except ConnectionClosed:
os.system("clear")
print _("pyLoad was terminated")
_exit(0)
except Exception, e:
println(2, red(str(e)))
self.cli.reset()
print_exc()
def print_help(config):
print
print "pyLoadCli Copyright (c) 2008-2012 the pyLoad Team"
print
print "Usage: [python] pyLoadCli.py [options] [command]"
print
print ""
print "See pyLoadCli.py -c for a complete listing."
print
print ""
print " -i, --interactive", " Start in interactive mode"
print
print " -u, --username=", " " * 2, "Specify user name"
print " --pw=", " " * 2, "Password"
print " -a, --address=", " " * 3, "Use address (current=%s)" % config["addr"]
print " -p, --port", " " * 7, "Use port (current=%s)" % config["port"]
print
print " -l, --language", " " * 3, "Set user interface language (current=%s)" % config["language"]
print " -h, --help", " " * 7, "Display this help text"
print " -c, --commands", " " * 3, "List all available commands"
print
def print_packages(data):
for pack in data:
print "Package %s (#%s):" % (pack.name, pack.pid)
for download in pack.links:
print "\t" + print_file(download)
print
def print_file(download):
return "#%(id)-6d %(name)-30s %(statusmsg)-10s %(plugin)-8s" % {
"id": download.fid,
"name": download.name,
"statusmsg": download.statusmsg,
"plugin": download.plugin
}
def print_status(download):
return "#%(id)-6s %(name)-40s Status: %(statusmsg)-10s Size: %(size)s" % {
"id": download.fid,
"name": download.name,
"statusmsg": download.statusmsg,
"size": download.format_size
}
def print_commands():
commands = [("status", _("Prints server status")),
("queue", _("Prints downloads in queue")),
("collector", _("Prints downloads in collector")),
("add ...", _("Adds package to queue")),
("add_coll ...", _("Adds package to collector")),
("del_file ...", _("Delete Files from Queue/Collector")),
("del_package ...", _("Delete Packages from Queue/Collector")),
("move ...", _("Move Packages from Queue to Collector or vice versa")),
("restart_file ...", _("Restart files")),
("restart_package ...", _("Restart packages")),
("check ...", _("Check online status, works with local container")),
("check_container path", _("Checks online status of a container file")),
("pause", _("Pause the server")),
("unpause", _("continue downloads")),
("toggle", _("Toggle pause/unpause")),
("kill", _("kill server")), ]
print _("List of commands:")
print
for c in commands:
print "%-35s %s" % c
def writeConfig(opts):
try:
with open(join(homedir, ".pyloadcli"), "w") as cfgfile:
cfgfile.write("[cli]")
for opt in opts:
cfgfile.write("%s=%s\n" % (opt, opts[opt]))
except:
print _("Couldn't write user config file")
def main():
config = {"addr": "127.0.0.1", "port": "7227", "language": "en"}
try:
config["language"] = os.environ["LANG"][0:2]
except:
pass
if (not exists(join(pypath, "locale", config["language"]))) or config["language"] == "":
config["language"] = "en"
configFile = ConfigParser.ConfigParser()
configFile.read(join(homedir, ".pyloadcli"))
if configFile.has_section("cli"):
for opt in configFile.items("cli"):
config[opt[0]] = opt[1]
gettext.setpaths([join(os.sep, "usr", "share", "pyload", "locale"), None])
translation = gettext.translation("pyLoadCli", join(pypath, "locale"),
languages=[config["language"],"en"],fallback=True)
translation.install(unicode=True)
interactive = False
command = None
username = ""
password = ""
shortOptions = 'iu:p:a:hcl:'
longOptions = ['interactive', "username=", "pw=", "address=", "port=", "help", "commands", "language="]
try:
opts, extraparams = getopt(sys.argv[1:], shortOptions, longOptions)
for option, params in opts:
if option in ("-i", "--interactive"):
interactive = True
elif option in ("-u", "--username"):
username = params
elif option in ("-a", "--address"):
config["addr"] = params
elif option in ("-p", "--port"):
config["port"] = params
elif option in ("-l", "--language"):
config["language"] = params
gettext.setpaths([join(os.sep, "usr", "share", "pyload", "locale"), None])
translation = gettext.translation("pyLoadCli", join(pypath, "locale"),
languages=[config["language"],"en"],fallback=True)
translation.install(unicode=True)
elif option in ("-h", "--help"):
print_help(config)
exit()
elif option in ("--pw"):
password = params
elif option in ("-c", "--comands"):
print_commands()
exit()
except GetoptError:
print 'Unknown Argument(s) "%s"' % " ".join(sys.argv[1:])
print_help(config)
exit()
if len(extraparams) >= 1:
command = extraparams
client = False
if interactive:
try:
client = ThriftClient(config["addr"], int(config["port"]), username, password)
except WrongLogin:
pass
except NoSSL:
print _("You need py-openssl to connect to this pyLoad core.")
exit()
except NoConnection:
config["addr"] = False
config["port"] = False
if not client:
if not config["addr"]: config["addr"] = raw_input(_("Address: "))
if not config["port"]: config["port"] = raw_input(_("Port: "))
if not username: username = raw_input(_("Username: "))
if not password:
from getpass import getpass
password = getpass(_("Password: "))
try:
client = ThriftClient(config["addr"], int(config["port"]), username, password)
except WrongLogin:
print _("Login data is wrong.")
except NoConnection:
print _("Could not establish connection to %(addr)s:%(port)s." % {"addr": config["addr"],
"port": config["port"]})
else:
try:
client = ThriftClient(config["addr"], int(config["port"]), username, password)
except WrongLogin:
print _("Login data is wrong.")
except NoConnection:
print _("Could not establish connection to %(addr)s:%(port)s." % {"addr": config["addr"],
"port": config["port"]})
except NoSSL:
print _("You need py-openssl to connect to this pyLoad core.")
if interactive and command: print _("Interactive mode ignored since you passed some commands.")
if client:
writeConfig(config)
cli = Cli(client, command)