X-Git-Url: https://code.grnet.gr/git/ganeti-local/blobdiff_plain/a14a17fc7a345dd6caa35ff17cd957a6e9e2d60b..68676a00a95239fcdd935758b7f6144cbec8b0fc:/lib/cli.py?ds=sidebyside diff --git a/lib/cli.py b/lib/cli.py index 31bc0b7..befcb2c 100644 --- a/lib/cli.py +++ b/lib/cli.py @@ -26,6 +26,7 @@ import sys import textwrap import os.path import copy +import time from cStringIO import StringIO from ganeti import utils @@ -40,7 +41,7 @@ from optparse import (OptionParser, make_option, TitledHelpFormatter, Option, OptionValueError, SUPPRESS_HELP) __all__ = ["DEBUG_OPT", "NOHDR_OPT", "SEP_OPT", "GenericMain", - "SubmitOpCode", "SubmitJob", "SubmitQuery", + "SubmitOpCode", "cli_option", "GenerateTable", "AskUser", "ARGS_NONE", "ARGS_FIXED", "ARGS_ATLEAST", "ARGS_ANY", "ARGS_ONE", "USEUNITS_OPT", "FIELDS_OPT", "FORCE_OPT", @@ -176,9 +177,6 @@ FIELDS_OPT = make_option("-o", "--output", dest="output", action="store", FORCE_OPT = make_option("-f", "--force", dest="force", action="store_true", default=False, help="Force the operation") -_LOCK_OPT = make_option("--lock-retries", default=None, - type="int", help=SUPPRESS_HELP) - TAG_SRC_OPT = make_option("--from", dest="tags_source", default=None, help="File with tag names") @@ -253,16 +251,15 @@ def _ParseArgs(argv, commands, aliases): "\n%(bin)s --help to see details, or" " man %(bin)s\n" % {"bin": binary}) # compute the max line length for cmd + usage - mlen = max([len(" %s %s" % (cmd, commands[cmd][3])) for cmd in commands]) + mlen = max([len(" %s" % cmd) for cmd in commands]) mlen = min(60, mlen) # should not get here... # and format a nice command list print "Commands:" for cmd in sortedcmds: - cmdstr = " %s %s" % (cmd, commands[cmd][3]) + cmdstr = " %s" % (cmd,) help_text = commands[cmd][4] help_lines = textwrap.wrap(help_text, 79-3-mlen) - print "%-*s - %s" % (mlen, cmdstr, - help_lines.pop(0)) + print "%-*s - %s" % (mlen, cmdstr, help_lines.pop(0)) for line in help_lines: print "%-*s %s" % (mlen, "", line) print @@ -282,7 +279,6 @@ def _ParseArgs(argv, commands, aliases): cmd = aliases[cmd] func, nargs, parser_opts, usage, description = commands[cmd] - parser_opts.append(_LOCK_OPT) parser = OptionParser(option_list=parser_opts, description=description, formatter=TitledHelpFormatter(), @@ -374,30 +370,39 @@ def AskUser(text, choices=None): def SubmitOpCode(op, proc=None, feedback_fn=None): - """Function to submit an opcode. + """Legacy function to submit an opcode. This is just a simple wrapper over the construction of the processor instance. It should be extended to better handle feedback and interaction functions. """ - if feedback_fn is None: - feedback_fn = logger.ToStdout - if proc is None: - proc = mcpu.Processor(feedback=feedback_fn) - return proc.ExecOpCode(op) - - -def SubmitJob(job, cl=None): - if cl is None: - cl = luxi.Client() - return cl.SubmitJob(job) - - -def SubmitQuery(data, cl=None): - if cl is None: - cl = luxi.Client() - return cl.Query(data) + # TODO: Fix feedback_fn situation. + cl = luxi.Client() + + job_id = cl.SubmitJob([op]) + + while True: + jobs = cl.QueryJobs([job_id], ["status"]) + if not jobs: + # job not found, go away! + raise errors.JobLost("Job with id %s lost" % job_id) + + # TODO: Handle canceled and archived jobs + status = jobs[0][0] + if status in (constants.JOB_STATUS_SUCCESS, constants.JOB_STATUS_ERROR): + break + time.sleep(1) + + jobs = cl.QueryJobs([job_id], ["status", "result"]) + if not jobs: + raise errors.JobLost("Job with id %s lost" % job_id) + + status, result = jobs[0] + if status == constants.JOB_STATUS_SUCCESS: + return result[0] + else: + raise errors.OpExecError(result) def FormatError(err): @@ -445,6 +450,15 @@ def FormatError(err): obuf.write("Failure: invalid tag(s) given:\n%s" % msg) elif isinstance(err, errors.GenericError): obuf.write("Unhandled Ganeti error: %s" % msg) + elif isinstance(err, luxi.NoMasterError): + obuf.write("Cannot communicate with the master daemon.\nIs it running" + " and listening on '%s'?" % err.args[0]) + elif isinstance(err, luxi.TimeoutError): + obuf.write("Timeout while talking to the master daemon. Error:\n" + "%s" % msg) + elif isinstance(err, luxi.ProtocolError): + obuf.write("Unhandled protocol error while talking to the master daemon:\n" + "%s" % msg) else: obuf.write("Unhandled exception: %s" % msg) return retcode, obuf.getvalue().rstrip('\n') @@ -485,17 +499,9 @@ def GenericMain(commands, override=None, aliases=None): for key, val in override.iteritems(): setattr(options, key, val) - logger.SetupLogging(debug=options.debug, program=binary) + logger.SetupLogging(program=binary, debug=options.debug) utils.debug = options.debug - try: - utils.Lock('cmd', max_retries=options.lock_retries, debug=options.debug) - except errors.LockError, err: - logger.ToStderr(str(err)) - return 1 - except KeyboardInterrupt: - logger.ToStderr("Aborting.") - return 1 if old_cmdline: logger.Info("run with arguments '%s'" % old_cmdline) @@ -503,14 +509,10 @@ def GenericMain(commands, override=None, aliases=None): logger.Info("run with no arguments") try: - try: - result = func(options, args) - except errors.GenericError, err: - result, err_msg = FormatError(err) - logger.ToStderr(err_msg) - finally: - utils.Unlock('cmd') - utils.LockCleanup() + result = func(options, args) + except (errors.GenericError, luxi.ProtocolError), err: + result, err_msg = FormatError(err) + logger.ToStderr(err_msg) return result