mirror of
				https://salsa.debian.org/freeipa-team/freeipa.git
				synced 2025-02-25 18:55:28 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			855 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			855 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Authors:
 | |
| #   Jason Gerard DeRose <jderose@redhat.com>
 | |
| #
 | |
| # Copyright (C) 2008  Red Hat
 | |
| # see file 'COPYING' for use and warranty information
 | |
| #
 | |
| # 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; version 2 only
 | |
| #
 | |
| # 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, write to the Free Software
 | |
| # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 | |
| 
 | |
| """
 | |
| Functionality for Command Line Interface.
 | |
| """
 | |
| 
 | |
| import re
 | |
| import textwrap
 | |
| import sys
 | |
| import getpass
 | |
| import code
 | |
| import optparse
 | |
| import socket
 | |
| import fcntl
 | |
| import termios
 | |
| import struct
 | |
| 
 | |
| import frontend
 | |
| import backend
 | |
| import errors
 | |
| import plugable
 | |
| import util
 | |
| from constants import CLI_TAB
 | |
| from parameters import Password
 | |
| 
 | |
| 
 | |
| def to_cli(name):
 | |
|     """
 | |
|     Takes a Python identifier and transforms it into form suitable for the
 | |
|     Command Line Interface.
 | |
|     """
 | |
|     assert isinstance(name, str)
 | |
|     return name.replace('_', '-')
 | |
| 
 | |
| 
 | |
| def from_cli(cli_name):
 | |
|     """
 | |
|     Takes a string from the Command Line Interface and transforms it into a
 | |
|     Python identifier.
 | |
|     """
 | |
|     return str(cli_name).replace('-', '_')
 | |
| 
 | |
| 
 | |
| class textui(backend.Backend):
 | |
|     """
 | |
|     Backend plugin to nicely format output to stdout.
 | |
|     """
 | |
| 
 | |
|     def get_tty_width(self):
 | |
|         """
 | |
|         Return the width (in characters) of output tty.
 | |
| 
 | |
|         If stdout is not a tty, this method will return ``None``.
 | |
|         """
 | |
|         # /usr/include/asm/termios.h says that struct winsize has four
 | |
|         # unsigned shorts, hence the HHHH
 | |
|         if sys.stdout.isatty():
 | |
|             try:
 | |
|                 winsize = fcntl.ioctl(sys.stdout, termios.TIOCGWINSZ,
 | |
|                                       struct.pack('HHHH', 0, 0, 0, 0))
 | |
|                 return struct.unpack('HHHH', winsize)[1]
 | |
|             except IOError:
 | |
|                 return None
 | |
| 
 | |
|     def max_col_width(self, rows, col=None):
 | |
|         """
 | |
|         Return the max width (in characters) of a specified column.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> rows = [
 | |
|         ...     ('a', 'package'),
 | |
|         ...     ('an', 'egg'),
 | |
|         ... ]
 | |
|         >>> ui.max_col_width(rows, col=0)  # len('an')
 | |
|         2
 | |
|         >>> ui.max_col_width(rows, col=1)  # len('package')
 | |
|         7
 | |
|         >>> ui.max_col_width(['a', 'cherry', 'py'])  # len('cherry')
 | |
|         6
 | |
|         """
 | |
|         if type(rows) not in (list, tuple):
 | |
|             raise TypeError(
 | |
|                 'rows: need %r or %r; got %r' % (list, tuple, rows)
 | |
|             )
 | |
|         if len(rows) == 0:
 | |
|             return 0
 | |
|         if col is None:
 | |
|             return max(len(row) for row in rows)
 | |
|         return max(len(row[col]) for row in rows)
 | |
| 
 | |
|     def __get_encoding(self, stream):
 | |
|         assert stream in (sys.stdin, sys.stdout)
 | |
|         if stream.encoding is None:
 | |
|             if stream.isatty():
 | |
|                 return sys.getdefaultencoding()
 | |
|             return 'UTF-8'
 | |
|         return stream.encoding
 | |
| 
 | |
|     def decode(self, str_buffer):
 | |
|         """
 | |
|         Decode text from stdin.
 | |
|         """
 | |
|         assert type(str_buffer) is str
 | |
|         encoding = self.__get_encoding(sys.stdin)
 | |
|         return str_buffer.decode(encoding)
 | |
| 
 | |
|     def encode(self, unicode_text):
 | |
|         """
 | |
|         Encode text for output to stdout.
 | |
|         """
 | |
|         assert type(unicode_text) is unicode
 | |
|         encoding = self.__get_encoding(sys.stdout)
 | |
|         return unicode_text.encode(encoding)
 | |
| 
 | |
|     def choose_number(self, n, singular, plural=None):
 | |
|         if n == 1 or plural is None:
 | |
|             return singular % n
 | |
|         return plural % n
 | |
| 
 | |
|     def print_plain(self, string):
 | |
|         """
 | |
|         Print exactly like ``print`` statement would.
 | |
|         """
 | |
|         print string
 | |
| 
 | |
|     def print_line(self, text, width=None):
 | |
|         """
 | |
|         Force printing on a single line, using ellipsis if needed.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_line('This line can fit!', width=18)
 | |
|         This line can fit!
 | |
|         >>> ui.print_line('This line wont quite fit!', width=18)
 | |
|         This line wont ...
 | |
| 
 | |
|         The above example aside, you normally should not specify the
 | |
|         ``width``.  When you don't, it is automatically determined by calling
 | |
|         `textui.get_tty_width()`.
 | |
|         """
 | |
|         if width is None:
 | |
|             width = self.get_tty_width()
 | |
|         if width is not None and width < len(text):
 | |
|             text = text[:width - 3] + '...'
 | |
|         print text
 | |
| 
 | |
|     def print_paragraph(self, text, width=None):
 | |
|         """
 | |
|         Print a paragraph, automatically word-wrapping to tty width.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> text = '''
 | |
|         ... Python is a dynamic object-oriented programming language that can
 | |
|         ... be used for many kinds of software development.
 | |
|         ... '''
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_paragraph(text, width=45)
 | |
|         Python is a dynamic object-oriented
 | |
|         programming language that can be used for
 | |
|         many kinds of software development.
 | |
| 
 | |
|         The above example aside, you normally should not specify the
 | |
|         ``width``.  When you don't, it is automatically determined by calling
 | |
|         `textui.get_tty_width()`.
 | |
| 
 | |
|         The word-wrapping is done using the Python ``textwrap`` module.  See:
 | |
| 
 | |
|             http://docs.python.org/library/textwrap.html
 | |
|         """
 | |
|         if width is None:
 | |
|             width = self.get_tty_width()
 | |
|         for line in textwrap.wrap(text.strip(), width):
 | |
|             print line
 | |
| 
 | |
|     def print_indented(self, text, indent=1):
 | |
|         """
 | |
|         Print at specified indentation level.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_indented('One indentation level.')
 | |
|           One indentation level.
 | |
|         >>> ui.print_indented('Two indentation levels.', indent=2)
 | |
|             Two indentation levels.
 | |
|         >>> ui.print_indented('No indentation.', indent=0)
 | |
|         No indentation.
 | |
|         """
 | |
|         print (CLI_TAB * indent + text)
 | |
| 
 | |
|     def print_keyval(self, rows, indent=1):
 | |
|         """
 | |
|         Print (key = value) pairs, one pair per line.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> items = [
 | |
|         ...     ('in_server', True),
 | |
|         ...     ('mode', 'production'),
 | |
|         ... ]
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_keyval(items)
 | |
|           in_server = True
 | |
|           mode = 'production'
 | |
|         >>> ui.print_keyval(items, indent=0)
 | |
|         in_server = True
 | |
|         mode = 'production'
 | |
| 
 | |
|         Also see `textui.print_indented`.
 | |
|         """
 | |
|         for (key, value) in rows:
 | |
|             self.print_indented('%s = %r' % (key, value), indent)
 | |
| 
 | |
|     def print_entry(self, entry, indent=1):
 | |
|         """
 | |
|         Print an ldap entry dict.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> entry = dict(sn='Last', givenname='First', uid='flast')
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_entry(entry)
 | |
|           givenname: 'First'
 | |
|           sn: 'Last'
 | |
|           uid: 'flast'
 | |
|         """
 | |
|         assert type(entry) is dict
 | |
|         for key in sorted(entry):
 | |
|             value = entry[key]
 | |
|             if type(value) in (list, tuple):
 | |
|                 value = ', '.join(repr(v) for v in value)
 | |
|             else:
 | |
|                 value = repr(value)
 | |
|             self.print_indented('%s: %s' % (key, value), indent)
 | |
| 
 | |
|     def print_dashed(self, string, above=True, below=True, indent=0, dash='-'):
 | |
|         """
 | |
|         Print a string with a dashed line above and/or below.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_dashed('Dashed above and below.')
 | |
|         -----------------------
 | |
|         Dashed above and below.
 | |
|         -----------------------
 | |
|         >>> ui.print_dashed('Only dashed below.', above=False)
 | |
|         Only dashed below.
 | |
|         ------------------
 | |
|         >>> ui.print_dashed('Only dashed above.', below=False)
 | |
|         ------------------
 | |
|         Only dashed above.
 | |
|         """
 | |
|         assert isinstance(dash, basestring)
 | |
|         assert len(dash) == 1
 | |
|         dashes = dash * len(string)
 | |
|         if above:
 | |
|             self.print_indented(dashes, indent)
 | |
|         self.print_indented(string, indent)
 | |
|         if below:
 | |
|             self.print_indented(dashes, indent)
 | |
| 
 | |
|     def print_h1(self, text):
 | |
|         """
 | |
|         Print a primary header at indentation level 0.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_h1('A primary header')
 | |
|         ================
 | |
|         A primary header
 | |
|         ================
 | |
|         """
 | |
|         self.print_dashed(text, indent=0, dash='=')
 | |
| 
 | |
|     def print_h2(self, text):
 | |
|         """
 | |
|         Print a secondary header at indentation level 1.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_h2('A secondary header')
 | |
|           ------------------
 | |
|           A secondary header
 | |
|           ------------------
 | |
|         """
 | |
|         self.print_dashed(text, indent=1, dash='-')
 | |
| 
 | |
|     def print_name(self, name):
 | |
|         """
 | |
|         Print a command name.
 | |
| 
 | |
|         The typical use for this is to mark the start of output from a
 | |
|         command.  For example, a hypothetical ``show_status`` command would
 | |
|         output something like this:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_name('show_status')
 | |
|         ------------
 | |
|         show-status:
 | |
|         ------------
 | |
|         """
 | |
|         self.print_dashed('%s:' % to_cli(name))
 | |
| 
 | |
|     def print_count(self, count, singular, plural=None):
 | |
|         """
 | |
|         Print a summary count.
 | |
| 
 | |
|         The typical use for this is to print the number of items returned
 | |
|         by a command, especially when this return count can vary.  This
 | |
|         preferably should be used as a summary and should be the final text
 | |
|         a command outputs.
 | |
| 
 | |
|         For example:
 | |
| 
 | |
|         >>> ui = textui()
 | |
|         >>> ui.print_count(1, '%d goose', '%d geese')
 | |
|         -------
 | |
|         1 goose
 | |
|         -------
 | |
|         >>> ui.print_count(['Don', 'Sue'], 'Found %d user', 'Found %d users')
 | |
|         -------------
 | |
|         Found 2 users
 | |
|         -------------
 | |
| 
 | |
|         If ``count`` is not an integer, it must be a list or tuple, and then
 | |
|         ``len(count)`` is used as the count.
 | |
|         """
 | |
|         if type(count) is not int:
 | |
|             assert type(count) in (list, tuple)
 | |
|             count = len(count)
 | |
|         self.print_dashed(
 | |
|             self.choose_number(count, singular, plural)
 | |
|         )
 | |
| 
 | |
|     def prompt(self, label, default=None, get_values=None):
 | |
|         """
 | |
|         Prompt user for input.
 | |
|         """
 | |
|         # TODO: Add tab completion using readline
 | |
|         if default is None:
 | |
|             prompt = u'%s: ' % label
 | |
|         else:
 | |
|             prompt = u'%s [%s]: ' % (label, default)
 | |
|         return self.decode(
 | |
|             raw_input(self.encode(prompt))
 | |
|         )
 | |
| 
 | |
|     def prompt_password(self, label):
 | |
|         """
 | |
|         Prompt user for a password.
 | |
|         """
 | |
|         try:
 | |
|             while True:
 | |
|                 pw1 = getpass.getpass('%s: ' % label)
 | |
|                 pw2 = getpass.getpass('Enter again to verify: ')
 | |
|                 if pw1 == pw2:
 | |
|                     return self.decode(pw1)
 | |
|                 print '  ** Passwords do not match. Please enter again. **'
 | |
|         except KeyboardInterrupt:
 | |
|             print ''
 | |
|             print '  ** Cancelled. **'
 | |
| 
 | |
| 
 | |
| class help(frontend.Application):
 | |
|     '''Display help on a command.'''
 | |
| 
 | |
|     takes_args = ['command?']
 | |
| 
 | |
|     def run(self, command):
 | |
|         textui = self.Backend.textui
 | |
|         if command is None:
 | |
|             self.print_commands()
 | |
|             return
 | |
|         key = str(command)
 | |
|         if key not in self.application:
 | |
|             raise errors.UnknownHelpError(key)
 | |
|         cmd = self.application[key]
 | |
|         print 'Purpose: %s' % cmd.doc
 | |
|         self.application.build_parser(cmd).print_help()
 | |
| 
 | |
|     def print_commands(self):
 | |
|         std = set(self.Command) - set(self.Application)
 | |
|         print '\nStandard IPA commands:'
 | |
|         for key in sorted(std):
 | |
|             cmd = self.api.Command[key]
 | |
|             self.print_cmd(cmd)
 | |
|         print '\nSpecial CLI commands:'
 | |
|         for cmd in self.api.Application():
 | |
|             self.print_cmd(cmd)
 | |
|         print '\nUse the --help option to see all the global options'
 | |
|         print ''
 | |
| 
 | |
|     def print_cmd(self, cmd):
 | |
|         print '  %s  %s' % (
 | |
|             to_cli(cmd.name).ljust(self.application.mcl),
 | |
|             cmd.doc,
 | |
|         )
 | |
| 
 | |
| 
 | |
| class console(frontend.Application):
 | |
|     """Start the IPA interactive Python console."""
 | |
| 
 | |
|     def run(self):
 | |
|         code.interact(
 | |
|             '(Custom IPA interactive Python console)',
 | |
|             local=dict(api=self.api)
 | |
|         )
 | |
| 
 | |
| 
 | |
| class show_api(frontend.Application):
 | |
|     'Show attributes on dynamic API object'
 | |
| 
 | |
|     takes_args = ('namespaces*',)
 | |
| 
 | |
|     def run(self, namespaces):
 | |
|         if namespaces is None:
 | |
|             names = tuple(self.api)
 | |
|         else:
 | |
|             for name in namespaces:
 | |
|                 if name not in self.api:
 | |
|                     raise errors.NoSuchNamespaceError(name)
 | |
|             names = namespaces
 | |
|         lines = self.__traverse(names)
 | |
|         ml = max(len(l[1]) for l in lines)
 | |
|         self.Backend.textui.print_name('run')
 | |
|         first = True
 | |
|         for line in lines:
 | |
|             if line[0] == 0 and not first:
 | |
|                 print ''
 | |
|             if first:
 | |
|                 first = False
 | |
|             print '%s%s %r' % (
 | |
|                 ' ' * line[0],
 | |
|                 line[1].ljust(ml),
 | |
|                 line[2],
 | |
|             )
 | |
|         if len(lines) == 1:
 | |
|             s = '1 attribute shown.'
 | |
|         else:
 | |
|             s = '%d attributes show.' % len(lines)
 | |
|         self.Backend.textui.print_dashed(s)
 | |
| 
 | |
| 
 | |
|     def __traverse(self, names):
 | |
|         lines = []
 | |
|         for name in names:
 | |
|             namespace = self.api[name]
 | |
|             self.__traverse_namespace('%s' % name, namespace, lines)
 | |
|         return lines
 | |
| 
 | |
|     def __traverse_namespace(self, name, namespace, lines, tab=0):
 | |
|         lines.append((tab, name, namespace))
 | |
|         for member_name in namespace:
 | |
|             member = namespace[member_name]
 | |
|             lines.append((tab + 1, member_name, member))
 | |
|             if not hasattr(member, '__iter__'):
 | |
|                 continue
 | |
|             for n in member:
 | |
|                 attr = member[n]
 | |
|                 if isinstance(attr, plugable.NameSpace) and len(attr) > 0:
 | |
|                     self.__traverse_namespace(n, attr, lines, tab + 2)
 | |
| 
 | |
| 
 | |
| cli_application_commands = (
 | |
|     help,
 | |
|     console,
 | |
|     show_api,
 | |
| )
 | |
| 
 | |
| 
 | |
| class KWCollector(object):
 | |
|     def __init__(self):
 | |
|         object.__setattr__(self, '_KWCollector__d', {})
 | |
| 
 | |
|     def __setattr__(self, name, value):
 | |
|         if name in self.__d:
 | |
|             v = self.__d[name]
 | |
|             if type(v) is tuple:
 | |
|                 value = v + (value,)
 | |
|             else:
 | |
|                 value = (v, value)
 | |
|         self.__d[name] = value
 | |
|         object.__setattr__(self, name, value)
 | |
| 
 | |
|     def __todict__(self):
 | |
|         return dict(self.__d)
 | |
| 
 | |
| 
 | |
| class CLI(object):
 | |
|     """
 | |
|     All logic for dispatching over command line interface.
 | |
|     """
 | |
| 
 | |
|     __d = None
 | |
|     __mcl = None
 | |
| 
 | |
|     def __init__(self, api, argv):
 | |
|         self.api = api
 | |
|         self.argv = tuple(argv)
 | |
|         self.__done = set()
 | |
| 
 | |
|     def run(self):
 | |
|         """
 | |
|         Call `CLI.run_real` in a try/except.
 | |
|         """
 | |
|         self.bootstrap()
 | |
|         try:
 | |
|             self.run_real()
 | |
|         except KeyboardInterrupt:
 | |
|             print ''
 | |
|             self.api.log.info('operation aborted')
 | |
|             sys.exit()
 | |
|         except errors.IPAError, e:
 | |
|             self.api.log.error(unicode(e))
 | |
|             sys.exit(e.faultCode)
 | |
| 
 | |
|     def run_real(self):
 | |
|         """
 | |
|         Parse ``argv`` and potentially run a command.
 | |
| 
 | |
|         This method requires several initialization steps to be completed
 | |
|         first, all of which all automatically called with a single call to
 | |
|         `CLI.finalize()`. The initialization steps are broken into separate
 | |
|         methods simply to make it easy to write unit tests.
 | |
| 
 | |
|         The initialization involves these steps:
 | |
| 
 | |
|             1. `CLI.parse_globals` parses the global options, which get stored
 | |
|                in ``CLI.options``, and stores the remaining args in
 | |
|                ``CLI.cmd_argv``.
 | |
| 
 | |
|             2. `CLI.bootstrap` initializes the environment information in
 | |
|                ``CLI.api.env``.
 | |
| 
 | |
|             3. `CLI.load_plugins` registers all plugins, including the
 | |
|                CLI-specific plugins.
 | |
| 
 | |
|             4. `CLI.finalize` instantiates all plugins and performs the
 | |
|                remaining initialization needed to use the `plugable.API`
 | |
|                instance.
 | |
|         """
 | |
|         self.__doing('run_real')
 | |
|         self.finalize()
 | |
|         if self.api.env.mode == 'unit_test':
 | |
|             return
 | |
|         if len(self.cmd_argv) < 1:
 | |
|             self.api.Command.help()
 | |
|             return
 | |
|         key = self.cmd_argv[0]
 | |
|         if key not in self:
 | |
|             raise errors.UnknownCommandError(key)
 | |
|         self.run_cmd(self[key])
 | |
| 
 | |
|         # FIXME: Stuff that might need special handling still:
 | |
| #        # Now run the command
 | |
| #        try:
 | |
| #            ret = cmd(**kw)
 | |
| #            if callable(cmd.output_for_cli):
 | |
| #                (args, options) = cmd.params_2_args_options(kw)
 | |
| #                cmd.output_for_cli(self.api.Backend.textui, ret, *args, **options)
 | |
| #            return 0
 | |
| #        except socket.error, e:
 | |
| #            print e[1]
 | |
| #            return 1
 | |
| #        except errors.GenericError, err:
 | |
| #            code = getattr(err,'faultCode',None)
 | |
| #            faultString = getattr(err,'faultString',None)
 | |
| #            if not code:
 | |
| #                raise err
 | |
| #            if code < errors.IPA_ERROR_BASE:
 | |
| #                print "%s: %s" % (code, faultString)
 | |
| #            else:
 | |
| #                print "%s: %s" % (code, getattr(err,'__doc__',''))
 | |
| #            return 1
 | |
| #        except StandardError, e:
 | |
| #            print e
 | |
| #            return 2
 | |
| 
 | |
|     def finalize(self):
 | |
|         """
 | |
|         Fully initialize ``CLI.api`` `plugable.API` instance.
 | |
| 
 | |
|         This method first calls `CLI.load_plugins` to perform some dependant
 | |
|         initialization steps, after which `plugable.API.finalize` is called.
 | |
| 
 | |
|         Finally, the CLI-specific commands are passed a reference to this
 | |
|         `CLI` instance by calling `frontend.Application.set_application`.
 | |
|         """
 | |
|         self.__doing('finalize')
 | |
|         self.load_plugins()
 | |
|         self.api.finalize()
 | |
|         for a in self.api.Application():
 | |
|             a.set_application(self)
 | |
|         assert self.__d is None
 | |
|         self.__d = dict(
 | |
|             (c.name.replace('_', '-'), c) for c in self.api.Command()
 | |
|         )
 | |
|         self.textui = self.api.Backend.textui
 | |
| 
 | |
|     def load_plugins(self):
 | |
|         """
 | |
|         Load all standard plugins plus the CLI-specific plugins.
 | |
| 
 | |
|         This method first calls `CLI.bootstrap` to preform some dependant
 | |
|         initialization steps, after which `plugable.API.load_plugins` is
 | |
|         called.
 | |
| 
 | |
|         Finally, all the CLI-specific plugins are registered.
 | |
|         """
 | |
|         self.__doing('load_plugins')
 | |
|         if 'bootstrap' not in self.__done:
 | |
|             self.bootstrap()
 | |
|         self.api.load_plugins()
 | |
|         for klass in cli_application_commands:
 | |
|             self.api.register(klass)
 | |
|         self.api.register(textui)
 | |
| 
 | |
|     def bootstrap(self):
 | |
|         """
 | |
|         Initialize the ``CLI.api.env`` environment variables.
 | |
| 
 | |
|         This method first calls `CLI.parse_globals` to perform some dependant
 | |
|         initialization steps. Then, using environment variables that may have
 | |
|         been passed in the global options, the ``overrides`` are constructed
 | |
|         and `plugable.API.bootstrap` is called.
 | |
|         """
 | |
|         self.__doing('bootstrap')
 | |
|         self.parse_globals()
 | |
|         self.api.bootstrap_with_global_options(self.options, context='cli')
 | |
| 
 | |
|     def parse_globals(self):
 | |
|         """
 | |
|         Parse out the global options.
 | |
| 
 | |
|         This method parses the global options out of the ``CLI.argv`` instance
 | |
|         attribute, after which two new instance attributes are available:
 | |
| 
 | |
|             1. ``CLI.options`` - an ``optparse.Values`` instance containing
 | |
|                the global options.
 | |
| 
 | |
|             2. ``CLI.cmd_argv`` - a tuple containing the remainder of
 | |
|                ``CLI.argv`` after the global options have been consumed.
 | |
| 
 | |
|         The common global options are added using the
 | |
|         `util.add_global_options` function.
 | |
|         """
 | |
|         self.__doing('parse_globals')
 | |
|         parser = optparse.OptionParser()
 | |
|         parser.disable_interspersed_args()
 | |
|         parser.add_option('-a', dest='prompt_all', action='store_true',
 | |
|                 help='Prompt for all missing options interactively')
 | |
|         parser.add_option('-n', dest='interactive', action='store_false',
 | |
|                 help='Don\'t prompt for any options interactively')
 | |
|         parser.set_defaults(
 | |
|             prompt_all=False,
 | |
|             interactive=True,
 | |
|         )
 | |
|         util.add_global_options(parser)
 | |
|         (options, args) = parser.parse_args(list(self.argv))
 | |
|         self.options = options
 | |
|         self.cmd_argv = tuple(args)
 | |
| 
 | |
|     def __doing(self, name):
 | |
|         if name in self.__done:
 | |
|             raise StandardError(
 | |
|                 '%s.%s() already called' % (self.__class__.__name__, name)
 | |
|             )
 | |
|         self.__done.add(name)
 | |
| 
 | |
|     def run_cmd(self, cmd):
 | |
|         kw = self.parse(cmd)
 | |
|         if self.options.interactive:
 | |
|             self.prompt_interactively(cmd, kw)
 | |
|         self.prompt_for_passwords(cmd, kw)
 | |
|         self.set_defaults(cmd, kw)
 | |
|         result = cmd(**kw)
 | |
|         if callable(cmd.output_for_cli):
 | |
|             for param in cmd.params():
 | |
|                 if isinstance(param, Password):
 | |
|                     try:
 | |
|                         del kw[param.name]
 | |
|                     except KeyError:
 | |
|                         pass
 | |
|             (args, options) = cmd.params_2_args_options(kw)
 | |
|             cmd.output_for_cli(self.api.Backend.textui, result, *args, **options)
 | |
| 
 | |
|     def set_defaults(self, cmd, kw):
 | |
|         for param in cmd.params():
 | |
|             if not kw.get(param.name):
 | |
|                 value = param.get_default(**kw)
 | |
|                 if value:
 | |
|                     kw[param.name] = value
 | |
| 
 | |
|     def prompt_for_passwords(self, cmd, kw):
 | |
|         for param in cmd.params():
 | |
|             if 'password' not in param.flags:
 | |
|                 continue
 | |
|             if kw.get(param.name, False) is True or param.name in cmd.args:
 | |
|                 kw[param.name] = self.textui.prompt_password(
 | |
|                     param.cli_name
 | |
|                 )
 | |
|             else:
 | |
|                 kw.pop(param.name, None)
 | |
|         return kw
 | |
| 
 | |
|     def prompt_interactively(self, cmd, kw):
 | |
|         """
 | |
|         Interactively prompt for missing or invalid values.
 | |
| 
 | |
|         By default this method will only prompt for *required* Param that
 | |
|         have a missing or invalid value.  However, if
 | |
|         ``CLI.options.prompt_all`` is True, this method will prompt for any
 | |
|         params that have a missing or required values, even if the param is
 | |
|         optional.
 | |
|         """
 | |
|         for param in cmd.params():
 | |
|             if 'password' in param.flags:
 | |
|                 continue
 | |
|             elif param.name not in kw:
 | |
|                 if not (param.required or self.options.prompt_all):
 | |
|                     continue
 | |
|                 default = param.get_default(**kw)
 | |
|                 error = None
 | |
|                 while True:
 | |
|                     if error is not None:
 | |
|                         print '>>> %s: %s' % (param.cli_name, error)
 | |
|                     raw = self.textui.prompt(param.cli_name, default)
 | |
|                     try:
 | |
|                         value = param(raw, **kw)
 | |
|                         if value is not None:
 | |
|                             kw[param.name] = value
 | |
|                         break
 | |
|                     except errors.ValidationError, e:
 | |
|                         error = e.error
 | |
|         return kw
 | |
| 
 | |
| # FIXME: This should be done as the plugins are loaded
 | |
| #        if self.api.env.server_context:
 | |
| #            try:
 | |
| #                import krbV
 | |
| #                import ldap
 | |
| #                from ipaserver import conn
 | |
| #                from ipaserver.servercore import context
 | |
| #                krbccache =  krbV.default_context().default_ccache().name
 | |
| #                context.conn = conn.IPAConn(self.api.env.ldaphost, self.api.env.ldapport, krbccache)
 | |
| #            except ImportError:
 | |
| #                print >> sys.stderr, "There was a problem importing a Python module: %s" % sys.exc_value
 | |
| #                return 2
 | |
| #            except ldap.LDAPError, e:
 | |
| #                print >> sys.stderr, "There was a problem connecting to the LDAP server: %s" % e[0].get('desc')
 | |
| #                return 2
 | |
| #        ret = cmd(**kw)
 | |
| #        if callable(cmd.output_for_cli):
 | |
| #            return cmd.output_for_cli(ret)
 | |
| #        else:
 | |
| #            return 0
 | |
| 
 | |
|     def parse(self, cmd):
 | |
|         parser = self.build_parser(cmd)
 | |
|         (kwc, args) = parser.parse_args(
 | |
|             list(self.cmd_argv[1:]), KWCollector()
 | |
|         )
 | |
|         kw = kwc.__todict__()
 | |
|         arg_kw = cmd.args_to_kw(*args)
 | |
|         assert set(arg_kw).intersection(kw) == set()
 | |
|         kw.update(arg_kw)
 | |
|         return kw
 | |
| 
 | |
|     def build_parser(self, cmd):
 | |
|         parser = optparse.OptionParser(
 | |
|             usage=self.get_usage(cmd),
 | |
|         )
 | |
|         for option in cmd.options():
 | |
|             kw = dict(
 | |
|                 dest=option.name,
 | |
|                 help=option.doc,
 | |
|             )
 | |
|             if 'password' in option.flags:
 | |
|                 kw['action'] = 'store_true'
 | |
|             elif option.type is bool:
 | |
|                 if option.default is True:
 | |
|                     kw['action'] = 'store_false'
 | |
|                 else:
 | |
|                     kw['action'] = 'store_true'
 | |
|             else:
 | |
|                 kw['metavar'] = metavar=option.__class__.__name__.upper()
 | |
|             o = optparse.make_option('--%s' % to_cli(option.cli_name), **kw)
 | |
|             parser.add_option(o)
 | |
|         return parser
 | |
| 
 | |
|     def get_usage(self, cmd):
 | |
|         return ' '.join(self.get_usage_iter(cmd))
 | |
| 
 | |
|     def get_usage_iter(self, cmd):
 | |
|         yield 'Usage: %%prog [global-options] %s' % to_cli(cmd.name)
 | |
|         for arg in cmd.args():
 | |
|             if 'password' in arg.flags:
 | |
|                 continue
 | |
|             name = to_cli(arg.cli_name).upper()
 | |
|             if arg.multivalue:
 | |
|                 name = '%s...' % name
 | |
|             if arg.required:
 | |
|                 yield name
 | |
|             else:
 | |
|                 yield '[%s]' % name
 | |
| 
 | |
|     def __get_mcl(self):
 | |
|         """
 | |
|         Returns the Max Command Length.
 | |
|         """
 | |
|         if self.__mcl is None:
 | |
|             if self.__d is None:
 | |
|                 return None
 | |
|             self.__mcl = max(len(k) for k in self.__d)
 | |
|         return self.__mcl
 | |
|     mcl = property(__get_mcl)
 | |
| 
 | |
|     def isdone(self, name):
 | |
|         """
 | |
|         Return True in method named ``name`` has already been called.
 | |
|         """
 | |
|         return name in self.__done
 | |
| 
 | |
|     def __contains__(self, key):
 | |
|         assert self.__d is not None, 'you must call finalize() first'
 | |
|         return key in self.__d
 | |
| 
 | |
|     def __getitem__(self, key):
 | |
|         assert self.__d is not None, 'you must call finalize() first'
 | |
|         return self.__d[key]
 |