summaryrefslogtreecommitdiffstats
path: root/ipalib/cli.py
diff options
context:
space:
mode:
Diffstat (limited to 'ipalib/cli.py')
-rw-r--r--ipalib/cli.py854
1 files changed, 854 insertions, 0 deletions
diff --git a/ipalib/cli.py b/ipalib/cli.py
new file mode 100644
index 00000000..fb2fd95f
--- /dev/null
+++ b/ipalib/cli.py
@@ -0,0 +1,854 @@
+# 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]