# Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreements. See the NOTICE file # distributed with this work for additional information # regarding copyright ownership. The ASF licenses this file # to you under the Apache License, Version 2.0 (the # "License"); you may not use this file except in compliance # with the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, # software distributed under the License is distributed on an # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY # KIND, either express or implied. See the License for the # specific language governing permissions and limitations # under the License. ''' Created on Aug 2, 2010 ''' import sys import os import inspect from optparse import OptionParser, OptParseError, BadOptionError, OptionError, OptionConflictError, OptionValueError import cloudapis as apis def describe(name,desc): def inner(decoratee): if not hasattr(decoratee,"descriptions"): decoratee.descriptions = {} decoratee.descriptions[name] = desc return decoratee return inner def error(msg): sys.stderr.write(msg) sys.stderr.write("\n") class MyOptionParser(OptionParser): def error(self, msg): error("%s: %s\n" % (self.get_prog_name(),msg)) self.print_usage(sys.stderr) self.exit(os.EX_USAGE) def parse_args(self,*args,**kwargs): options,arguments = OptionParser.parse_args(self,*args,**kwargs) def prune_options(options,alist): """Given 'options' -- a list of arguments to OptionParser.add_option, and a set of optparse Values, return a dictionary of only those values that apply exclusively to 'options'""" return dict( [ (k,getattr(options,k)) for k in dir(options) if k in alist ] ) api_options = prune_options(options,self.api_dests) cmd_options = prune_options(options,self.cmd_dests) return options,arguments,api_options,cmd_options def get_parser(api_callable=None,cmd_callable=None): # this should probably be the __init__ method of myoptionparser def getdefaulttag(default): if default is not None: return " [Default: %default]" return '' def get_arguments_and_options(callable): """Infers and returns arguments and options based on a callable's signature. Cooperates with decorator @describe""" try: funcargs = inspect.getargspec(callable).args defaults = inspect.getargspec(callable).defaults except: funcargs = inspect.getargspec(callable)[0] defaults = inspect.getargspec(callable)[3] if not defaults: defaults = [] args = funcargs[1:len(funcargs)-len(defaults)] # this assumes self, so assumes methods opts = funcargs[len(funcargs)-len(defaults):] try: descriptions = callable.descriptions except AttributeError: descriptions = {} arguments = [ (argname, descriptions.get(argname,'') ) for argname in args ] options = [ [ ("--%s"%argname.replace("_","-"),), { "dest":argname, "help":descriptions.get(argname,'') + getdefaulttag(default), "default":default, } ] for argname,default in zip(opts,defaults) ] return arguments,options basic_usage = "usage: %prog [options...] " api_name = "" cmd_name = "" description = "%prog is a command-line tool to access several cloud APIs." arguments = '' argexp = "" if api_callable: api_name = api_callable.__module__.split(".")[-1].replace("_","-") api_arguments,api_options = get_arguments_and_options(api_callable) assert len(api_arguments) is 0 # no mandatory arguments for class initializers if cmd_callable: cmd_name = cmd_callable.func_name.replace("_","-") cmd_arguments,cmd_options = get_arguments_and_options(cmd_callable) if cmd_arguments: arguments = " " + " ".join( [ s[0].upper() for s in cmd_arguments ] ) argexp = "\n\nArguments:\n" + "\n".join ( " %s\n %s"%(s.upper(),u) for s,u in cmd_arguments ) description = cmd_callable.__doc__ api_command = "%s %s"%(api_name,cmd_name) if description: description = "\n\n" + description else: description = '' usage = basic_usage + api_command + arguments + description + argexp parser = MyOptionParser(usage=usage, add_help_option=False) parser.add_option('--help', action="help") group = parser.add_option_group("General options") group.add_option('-v', '--verbose', dest="verbose", help="Print extra output") parser.api_dests = [] if api_callable and api_options: group = parser.add_option_group("Options for the %s API"%api_name) for a in api_options: group.add_option(a[0][0],**a[1]) parser.api_dests.append(a[1]["dest"]) parser.cmd_dests = [] if cmd_callable and cmd_options: group = parser.add_option_group("Options for the %s command"%cmd_name) for a in cmd_options: group.add_option(a[0][0],**a[1]) parser.cmd_dests.append(a[1]["dest"]) return parser def lookup_command_in_api(api,command_name): command = getattr(api,command_name.replace("-","_"),None) return command def get_api_list(api): apilist = [] for cmd_name in dir(api): cmd = getattr(api,cmd_name) if callable(cmd) and not cmd_name.startswith("_"): apilist.append(cmd_name) return apilist def get_command_list(api): cmds = [] for cmd_name in dir(api): cmd = getattr(api,cmd_name) if callable(cmd) and not cmd_name.startswith("_"): if cmd.__doc__:docstring = cmd.__doc__ else:docstring = '' cmds.append( " %s" % (cmd_name.replace('_','-')) ) return cmds