2008-08-08 16:46:23 -05:00
|
|
|
# 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
|
|
|
|
|
|
|
|
"""
|
2008-09-23 19:01:29 -05:00
|
|
|
Functionality for Command Line Interface.
|
2008-08-08 16:46:23 -05:00
|
|
|
"""
|
|
|
|
|
2008-08-11 14:35:57 -05:00
|
|
|
import re
|
2008-11-12 01:46:04 -06:00
|
|
|
import textwrap
|
2008-08-11 21:03:47 -05:00
|
|
|
import sys
|
2008-11-18 12:30:16 -06:00
|
|
|
import getpass
|
2008-08-26 19:25:33 -05:00
|
|
|
import code
|
2008-09-04 01:33:57 -05:00
|
|
|
import optparse
|
2008-11-03 16:38:05 -06:00
|
|
|
import socket
|
2008-10-17 15:55:03 -05:00
|
|
|
|
2008-09-23 19:01:29 -05:00
|
|
|
import frontend
|
2008-11-12 01:46:04 -06:00
|
|
|
import backend
|
2008-09-04 03:16:12 -05:00
|
|
|
import errors
|
2008-09-08 16:37:02 -05:00
|
|
|
import plugable
|
2008-09-08 20:41:15 -05:00
|
|
|
import ipa_types
|
2008-10-17 15:55:03 -05:00
|
|
|
from config import set_default_env, read_config
|
2008-10-31 19:17:08 -05:00
|
|
|
import util
|
2008-11-12 01:46:04 -06:00
|
|
|
from constants import CLI_TAB
|
2008-09-10 18:33:36 -05:00
|
|
|
|
|
|
|
|
2008-08-08 16:46:23 -05:00
|
|
|
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.
|
|
|
|
"""
|
2008-08-13 20:09:11 -05:00
|
|
|
return str(cli_name).replace('-', '_')
|
2008-08-11 14:35:57 -05:00
|
|
|
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
class textui(backend.Backend):
|
2008-09-24 00:03:10 -05:00
|
|
|
"""
|
2008-11-12 01:46:04 -06:00
|
|
|
Backend plugin to nicely format output to stdout.
|
2008-09-24 00:03:10 -05:00
|
|
|
"""
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
def get_tty_width(self):
|
|
|
|
"""
|
|
|
|
Return the width (in characters) of output tty.
|
|
|
|
|
|
|
|
If stdout is not a tty, this method will return ``None``.
|
|
|
|
"""
|
|
|
|
if sys.stdout.isatty():
|
|
|
|
return 80 # FIXME: we need to return the actual tty width
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
2008-11-19 04:31:29 -06:00
|
|
|
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)
|
|
|
|
|
2008-11-14 14:33:42 -06:00
|
|
|
def choose_number(self, n, singular, plural=None):
|
|
|
|
if n == 1 or plural is None:
|
|
|
|
return singular % n
|
|
|
|
return plural % n
|
2008-11-12 01:46:04 -06:00
|
|
|
|
2008-11-14 14:33:42 -06:00
|
|
|
def print_plain(self, string):
|
|
|
|
"""
|
|
|
|
Print exactly like ``print`` statement would.
|
2008-11-12 01:46:04 -06:00
|
|
|
"""
|
2008-09-24 00:03:10 -05:00
|
|
|
print string
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
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
|
|
|
|
|
2008-11-14 14:33:42 -06:00
|
|
|
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
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
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`.
|
|
|
|
"""
|
2008-11-14 23:03:31 -06:00
|
|
|
for (key, value) in rows:
|
|
|
|
self.print_indented('%s = %r' % (key, value), indent)
|
2008-11-12 01:46:04 -06:00
|
|
|
|
2008-11-17 16:27:08 -06:00
|
|
|
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)
|
|
|
|
|
2008-11-17 21:41:01 -06:00
|
|
|
def print_dashed(self, string, above=True, below=True, indent=0, dash='-'):
|
2008-11-14 14:33:42 -06:00
|
|
|
"""
|
|
|
|
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.
|
|
|
|
"""
|
2008-11-17 21:41:01 -06:00
|
|
|
assert isinstance(dash, basestring)
|
|
|
|
assert len(dash) == 1
|
|
|
|
dashes = dash * len(string)
|
2008-11-14 14:33:42 -06:00
|
|
|
if above:
|
2008-11-17 21:41:01 -06:00
|
|
|
self.print_indented(dashes, indent)
|
|
|
|
self.print_indented(string, indent)
|
2008-11-14 14:33:42 -06:00
|
|
|
if below:
|
2008-11-17 21:41:01 -06:00
|
|
|
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='-')
|
2008-11-14 14:33:42 -06:00
|
|
|
|
|
|
|
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))
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
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)
|
|
|
|
)
|
|
|
|
|
2008-11-14 00:54:34 -06:00
|
|
|
def prompt(self, label, default=None, get_values=None):
|
|
|
|
"""
|
|
|
|
Prompt user for input.
|
|
|
|
"""
|
2008-11-19 04:31:29 -06:00
|
|
|
# TODO: Add tab completion using readline
|
2008-11-14 00:54:34 -06:00
|
|
|
if default is None:
|
2008-11-19 04:31:29 -06:00
|
|
|
prompt = u'%s: ' % label
|
2008-11-14 00:54:34 -06:00
|
|
|
else:
|
2008-11-19 04:31:29 -06:00
|
|
|
prompt = u'%s [%s]: ' % (label, default)
|
|
|
|
return self.decode(
|
|
|
|
raw_input(self.encode(prompt))
|
|
|
|
)
|
2008-11-14 00:54:34 -06:00
|
|
|
|
2008-11-18 12:30:16 -06:00
|
|
|
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:
|
2008-11-19 04:31:29 -06:00
|
|
|
return self.decode(pw1)
|
2008-11-18 12:30:16 -06:00
|
|
|
print ' ** Passwords do not match. Please enter again. **'
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
print ''
|
|
|
|
print ' ** Cancelled. **'
|
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
|
2008-09-23 19:01:29 -05:00
|
|
|
class help(frontend.Application):
|
2008-11-12 11:15:24 -06:00
|
|
|
'''Display help on a command.'''
|
2008-09-08 20:41:15 -05:00
|
|
|
|
2008-11-12 11:15:24 -06:00
|
|
|
takes_args = ['command?']
|
2008-09-08 20:41:15 -05:00
|
|
|
|
2008-11-12 11:15:24 -06:00
|
|
|
def run(self, command):
|
|
|
|
textui = self.Backend.textui
|
|
|
|
if command is None:
|
|
|
|
self.print_commands()
|
|
|
|
return
|
|
|
|
key = str(command)
|
2008-09-04 00:18:14 -05:00
|
|
|
if key not in self.application:
|
2008-11-14 00:29:35 -06:00
|
|
|
raise errors.UnknownHelpError(key)
|
2008-09-04 00:18:14 -05:00
|
|
|
cmd = self.application[key]
|
|
|
|
print 'Purpose: %s' % cmd.doc
|
2008-09-04 01:33:57 -05:00
|
|
|
self.application.build_parser(cmd).print_help()
|
2008-08-12 23:02:39 -05:00
|
|
|
|
2008-11-12 11:15:24 -06:00
|
|
|
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,
|
|
|
|
)
|
|
|
|
|
2008-10-27 15:48:02 -05:00
|
|
|
|
2008-09-23 19:01:29 -05:00
|
|
|
class console(frontend.Application):
|
2008-11-12 01:46:04 -06:00
|
|
|
"""Start the IPA interactive Python console."""
|
2008-08-26 19:25:33 -05:00
|
|
|
|
2008-09-23 22:10:35 -05:00
|
|
|
def run(self):
|
2008-08-26 19:25:33 -05:00
|
|
|
code.interact(
|
2008-09-04 00:18:14 -05:00
|
|
|
'(Custom IPA interactive Python console)',
|
2008-08-26 19:25:33 -05:00
|
|
|
local=dict(api=self.api)
|
|
|
|
)
|
|
|
|
|
2008-09-23 22:10:35 -05:00
|
|
|
|
2008-11-12 01:46:04 -06:00
|
|
|
class show_api(frontend.Application):
|
2008-09-24 00:35:40 -05:00
|
|
|
'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:
|
|
|
|
exit_error('api has no such namespace: %s' % name)
|
|
|
|
names = namespaces
|
|
|
|
lines = self.__traverse(names)
|
2008-09-08 16:37:02 -05:00
|
|
|
ml = max(len(l[1]) for l in lines)
|
2008-09-24 00:35:40 -05:00
|
|
|
self.print_name()
|
|
|
|
first = True
|
2008-09-08 16:37:02 -05:00
|
|
|
for line in lines:
|
2008-09-24 00:35:40 -05:00
|
|
|
if line[0] == 0 and not first:
|
2008-09-08 16:37:02 -05:00
|
|
|
print ''
|
2008-09-24 00:35:40 -05:00
|
|
|
if first:
|
|
|
|
first = False
|
2008-09-08 16:37:02 -05:00
|
|
|
print '%s%s %r' % (
|
|
|
|
' ' * line[0],
|
|
|
|
line[1].ljust(ml),
|
|
|
|
line[2],
|
|
|
|
)
|
2008-09-24 00:35:40 -05:00
|
|
|
if len(lines) == 1:
|
|
|
|
s = '1 attribute shown.'
|
|
|
|
else:
|
|
|
|
s = '%d attributes show.' % len(lines)
|
|
|
|
self.print_dashed(s)
|
|
|
|
|
2008-09-08 16:37:02 -05:00
|
|
|
|
2008-09-24 00:35:40 -05:00
|
|
|
def __traverse(self, names):
|
2008-09-08 16:37:02 -05:00
|
|
|
lines = []
|
2008-09-24 00:35:40 -05:00
|
|
|
for name in names:
|
2008-09-08 16:37:02 -05:00
|
|
|
namespace = self.api[name]
|
2008-09-24 00:35:40 -05:00
|
|
|
self.__traverse_namespace('%s' % name, namespace, lines)
|
2008-09-08 16:37:02 -05:00
|
|
|
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]
|
2008-09-08 16:44:53 -05:00
|
|
|
if isinstance(attr, plugable.NameSpace) and len(attr) > 0:
|
2008-09-08 16:37:02 -05:00
|
|
|
self.__traverse_namespace(n, attr, lines, tab + 2)
|
|
|
|
|
2008-08-26 19:25:33 -05:00
|
|
|
|
2008-09-23 23:44:52 -05:00
|
|
|
cli_application_commands = (
|
|
|
|
help,
|
|
|
|
console,
|
2008-09-24 00:35:40 -05:00
|
|
|
show_api,
|
2008-09-23 23:44:52 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2008-09-04 01:33:57 -05:00
|
|
|
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)
|
|
|
|
|
|
|
|
|
2008-08-11 14:35:57 -05:00
|
|
|
class CLI(object):
|
2008-10-27 20:21:49 -05:00
|
|
|
"""
|
|
|
|
All logic for dispatching over command line interface.
|
|
|
|
"""
|
|
|
|
|
2008-08-12 18:33:02 -05:00
|
|
|
__d = None
|
2008-08-12 20:52:17 -05:00
|
|
|
__mcl = None
|
2008-08-12 18:33:02 -05:00
|
|
|
|
2008-10-27 15:48:02 -05:00
|
|
|
def __init__(self, api, argv):
|
|
|
|
self.api = api
|
|
|
|
self.argv = tuple(argv)
|
|
|
|
self.__done = set()
|
2008-08-11 14:35:57 -05:00
|
|
|
|
2008-11-14 00:29:35 -06:00
|
|
|
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):
|
2008-10-27 20:21:49 -05:00
|
|
|
"""
|
2008-10-28 00:30:55 -05:00
|
|
|
Parse ``argv`` and potentially run a command.
|
2008-10-27 20:21:49 -05:00
|
|
|
|
|
|
|
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.
|
|
|
|
"""
|
2008-11-14 00:29:35 -06:00
|
|
|
self.__doing('run_real')
|
2008-10-27 20:21:49 -05:00
|
|
|
self.finalize()
|
2008-10-28 00:39:43 -05:00
|
|
|
if self.api.env.mode == 'unit_test':
|
2008-10-28 00:30:55 -05:00
|
|
|
return
|
2008-10-27 20:21:49 -05:00
|
|
|
if len(self.cmd_argv) < 1:
|
2008-11-14 00:29:35 -06:00
|
|
|
self.api.Command.help()
|
|
|
|
return
|
2008-10-27 20:21:49 -05:00
|
|
|
key = self.cmd_argv[0]
|
|
|
|
if key not in self:
|
2008-11-14 00:29:35 -06:00
|
|
|
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
|
2008-10-27 20:21:49 -05:00
|
|
|
|
|
|
|
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()
|
|
|
|
)
|
2008-11-14 00:54:34 -06:00
|
|
|
self.textui = self.api.Backend.textui
|
2008-10-27 20:21:49 -05:00
|
|
|
|
|
|
|
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')
|
2008-11-14 00:29:35 -06:00
|
|
|
if 'bootstrap' not in self.__done:
|
|
|
|
self.bootstrap()
|
2008-10-27 20:21:49 -05:00
|
|
|
self.api.load_plugins()
|
|
|
|
for klass in cli_application_commands:
|
|
|
|
self.api.register(klass)
|
2008-11-12 01:46:04 -06:00
|
|
|
self.api.register(textui)
|
2008-10-27 20:21:49 -05:00
|
|
|
|
|
|
|
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()
|
2008-10-31 20:03:07 -05:00
|
|
|
self.api.bootstrap_with_global_options(self.options, context='cli')
|
2008-10-27 20:21:49 -05:00
|
|
|
|
|
|
|
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.
|
2008-10-31 20:03:07 -05:00
|
|
|
|
|
|
|
The common global options are added using the
|
|
|
|
`util.add_global_options` function.
|
2008-10-27 20:21:49 -05:00
|
|
|
"""
|
|
|
|
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,
|
|
|
|
)
|
2008-10-31 19:17:08 -05:00
|
|
|
util.add_global_options(parser)
|
2008-10-27 20:21:49 -05:00
|
|
|
(options, args) = parser.parse_args(list(self.argv))
|
|
|
|
self.options = options
|
|
|
|
self.cmd_argv = tuple(args)
|
|
|
|
|
2008-10-27 15:48:02 -05:00
|
|
|
def __doing(self, name):
|
|
|
|
if name in self.__done:
|
|
|
|
raise StandardError(
|
|
|
|
'%s.%s() already called' % (self.__class__.__name__, name)
|
|
|
|
)
|
|
|
|
self.__done.add(name)
|
2008-08-11 14:35:57 -05:00
|
|
|
|
2008-10-28 00:30:55 -05:00
|
|
|
def run_cmd(self, cmd):
|
|
|
|
kw = self.parse(cmd)
|
|
|
|
if self.options.interactive:
|
2008-11-18 14:43:43 -06:00
|
|
|
self.prompt_interactively(cmd, kw)
|
|
|
|
self.prompt_for_passwords(cmd, kw)
|
2008-11-14 00:29:35 -06:00
|
|
|
result = cmd(**kw)
|
|
|
|
if callable(cmd.output_for_cli):
|
2008-11-18 14:43:43 -06:00
|
|
|
for param in cmd.params():
|
|
|
|
if param.ispassword():
|
|
|
|
del kw[param.name]
|
2008-11-14 00:29:35 -06:00
|
|
|
(args, options) = cmd.params_2_args_options(kw)
|
|
|
|
cmd.output_for_cli(self.api.Backend.textui, result, *args, **options)
|
2008-09-04 02:18:26 -05:00
|
|
|
|
2008-11-18 14:43:43 -06:00
|
|
|
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
|
|
|
|
|
2008-10-28 00:30:55 -05:00
|
|
|
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.
|
|
|
|
"""
|
2008-09-21 14:00:41 -05:00
|
|
|
for param in cmd.params():
|
2008-11-18 12:30:16 -06:00
|
|
|
if 'password' in param.flags:
|
2008-11-18 14:43:43 -06:00
|
|
|
continue
|
2008-11-18 12:30:16 -06:00
|
|
|
elif param.name not in kw:
|
2008-10-28 00:30:55 -05:00
|
|
|
if not (param.required or self.options.prompt_all):
|
|
|
|
continue
|
2008-09-21 14:00:41 -05:00
|
|
|
default = param.get_default(**kw)
|
2008-09-04 02:18:26 -05:00
|
|
|
error = None
|
|
|
|
while True:
|
|
|
|
if error is not None:
|
2008-10-13 23:00:18 -05:00
|
|
|
print '>>> %s: %s' % (param.cli_name, error)
|
2008-11-14 00:54:34 -06:00
|
|
|
raw = self.textui.prompt(param.cli_name, default)
|
2008-09-04 03:16:12 -05:00
|
|
|
try:
|
2008-09-21 14:00:41 -05:00
|
|
|
value = param(raw, **kw)
|
2008-09-04 03:16:12 -05:00
|
|
|
if value is not None:
|
2008-09-21 14:00:41 -05:00
|
|
|
kw[param.name] = value
|
2008-09-04 02:18:26 -05:00
|
|
|
break
|
2008-09-04 03:16:12 -05:00
|
|
|
except errors.ValidationError, e:
|
|
|
|
error = e.error
|
2008-10-28 00:30:55 -05:00
|
|
|
return kw
|
2008-09-04 01:33:57 -05:00
|
|
|
|
2008-10-28 00:30:55 -05:00
|
|
|
# FIXME: This should be done as the plugins are loaded
|
|
|
|
# if self.api.env.server_context:
|
|
|
|
# try:
|
|
|
|
# import krbV
|
|
|
|
# import ldap
|
|
|
|
# from ipa_server import conn
|
|
|
|
# from ipa_server.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):
|
2008-09-04 01:33:57 -05:00
|
|
|
parser = self.build_parser(cmd)
|
2008-10-28 00:30:55 -05:00
|
|
|
(kwc, args) = parser.parse_args(
|
2008-10-28 02:39:02 -05:00
|
|
|
list(self.cmd_argv[1:]), KWCollector()
|
2008-10-28 00:30:55 -05:00
|
|
|
)
|
2008-09-21 14:00:41 -05:00
|
|
|
kw = kwc.__todict__()
|
2008-11-18 12:30:16 -06:00
|
|
|
arg_kw = cmd.args_to_kw(*args)
|
2008-09-21 14:00:41 -05:00
|
|
|
assert set(arg_kw).intersection(kw) == set()
|
|
|
|
kw.update(arg_kw)
|
|
|
|
return kw
|
2008-09-04 01:33:57 -05:00
|
|
|
|
|
|
|
def build_parser(self, cmd):
|
|
|
|
parser = optparse.OptionParser(
|
2008-09-08 20:41:15 -05:00
|
|
|
usage=self.get_usage(cmd),
|
2008-09-04 01:33:57 -05:00
|
|
|
)
|
2008-09-10 10:31:34 -05:00
|
|
|
for option in cmd.options():
|
2008-11-18 12:30:16 -06:00
|
|
|
kw = dict(
|
2008-10-13 21:31:10 -05:00
|
|
|
dest=option.name,
|
2008-09-04 01:33:57 -05:00
|
|
|
help=option.doc,
|
|
|
|
)
|
2008-11-18 12:30:16 -06:00
|
|
|
if 'password' in option.flags:
|
|
|
|
kw['action'] = 'store_true'
|
|
|
|
elif isinstance(option.type, ipa_types.Bool):
|
2008-11-17 19:50:30 -06:00
|
|
|
if option.default is True:
|
2008-11-18 12:30:16 -06:00
|
|
|
kw['action'] = 'store_false'
|
2008-11-17 19:50:30 -06:00
|
|
|
else:
|
2008-11-18 12:30:16 -06:00
|
|
|
kw['action'] = 'store_true'
|
|
|
|
else:
|
|
|
|
kw['metavar'] = metavar=option.type.name.upper()
|
|
|
|
o = optparse.make_option('--%s' % to_cli(option.cli_name), **kw)
|
2008-10-21 08:31:06 -05:00
|
|
|
parser.add_option(o)
|
2008-09-04 01:33:57 -05:00
|
|
|
return parser
|
|
|
|
|
2008-09-08 20:41:15 -05:00
|
|
|
def get_usage(self, cmd):
|
|
|
|
return ' '.join(self.get_usage_iter(cmd))
|
|
|
|
|
|
|
|
def get_usage_iter(self, cmd):
|
2008-10-03 15:13:50 -05:00
|
|
|
yield 'Usage: %%prog [global-options] %s' % to_cli(cmd.name)
|
2008-09-09 21:02:26 -05:00
|
|
|
for arg in cmd.args():
|
2008-11-18 14:43:43 -06:00
|
|
|
if 'password' in arg.flags:
|
|
|
|
continue
|
2008-10-13 21:31:10 -05:00
|
|
|
name = to_cli(arg.cli_name).upper()
|
2008-09-08 20:41:15 -05:00
|
|
|
if arg.multivalue:
|
|
|
|
name = '%s...' % name
|
|
|
|
if arg.required:
|
|
|
|
yield name
|
|
|
|
else:
|
|
|
|
yield '[%s]' % name
|
|
|
|
|
2008-08-12 20:52:17 -05:00
|
|
|
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)
|
2008-10-28 00:30:55 -05:00
|
|
|
|
|
|
|
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]
|