1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
# -*- coding: UTF-8 -*-
# Copyright 2014 Red Hat, Inc.
# Part of clufter project
# Licensed under GPLv2 (a copy included | http://gnu.org/licenses/gpl-2.0.txt)
"""Command manager"""
__author__ = "Jan Pokorný <jpokorny @at@ Red Hat .dot. com>"
import logging
from .command import commands
from .error import ClufterError, ClufterPlainError, \
EC
from .plugin_registry import PluginManager
from .utils import apply_preserving_depth, \
apply_aggregation_preserving_depth, \
apply_intercalate, \
make_options
log = logging.getLogger(__name__)
class CommandManagerError(ClufterError):
pass
class CommandNotFoundError(ClufterPlainError):
def __init__(self, cmd):
super(CommandNotFoundError, self).__init__("Command not found: `{0}'",
cmd)
class CommandManager(PluginManager):
"""Class responsible for commands routing to filters or other actions"""
_default_registry = commands
def _init_handle_plugins(self, commands, flt_mgr):
log.debug("Commands before resolving: {0}"
.format(commands))
self._commands = self._resolve(flt_mgr.filters, commands)
def __iter__(self):
return self._commands.itervalues()
@staticmethod
def _resolve(filters, commands):
for cmd_name, cmd_cls in commands.items():
res_input = cmd_cls.filter_chain
res_output = apply_preserving_depth(filters.get)(res_input)
if apply_aggregation_preserving_depth(all)(res_output):
log.debug("resolve at `{0}' command: `{1}' -> {2}"
.format(cmd_name, repr(res_input), repr(res_output)))
commands[cmd_name] = cmd_cls(*res_output)
continue
# drop the command if cannot resolve any of the filters
res_input = apply_intercalate(res_input)
log.debug("cmd_name {0}".format(res_input))
map(lambda (i, x): log.warning("Resolve at `{0}' command:"
" `{1}' (#{2}) filter fail"
.format(cmd_name, res_input[i], i)),
filter(lambda (i, x): not(x),
enumerate(apply_intercalate(res_output))))
commands.pop(cmd_name)
return commands
@property
def commands(self):
return self._commands.copy()
def completion(self, completion):
return completion(iter(self))
def __call__(self, parser, args=None):
"""Follow up of the entry point, facade to particular commands"""
ec = EC.EXIT_SUCCESS
values = parser.values
try:
cmd = getattr(values, 'help', None) or args[0]
command = self._commands.get(cmd, None)
if not command:
raise CommandNotFoundError(cmd)
canonical_cmd = command.__class__.name
parser.description, options = command.parser_desc_opts
parser.option_groups[0].add_options(make_options(options))
args = ['--help'] if values.help else args[1:]
parser.defaults.update(values.__dict__) # from global options
opts, args = parser.parse_args(args)
if opts.help:
usage = '\n'.join(map(
lambda c:
"%prog [<global option> ...] {0} [<cmd option ...>]"
.format(c),
sorted(set([cmd, canonical_cmd]),
key=lambda i: int(i == canonical_cmd))
))
print parser.format_customized_help(usage=usage)
return ec
logging.getLogger().setLevel(opts.loglevel)
log.debug("Running command `{0}'; opts={1}, args={2}"
.format(cmd, opts.__dict__, args))
ec = command(opts, args)
except ClufterError as e:
ec = EC.EXIT_FAILURE
print e
if isinstance(e, CommandNotFoundError):
print "\nSupported commands:\n" + self.cmds()
#except Exception as e:
# print "OOPS: underlying unexpected exception:\n{0}".format(e)
# ec = EC.EXIT_FAILURE
return ec
def cmds(self, ind='', sep='\n'):
"""Return string containing formatted list of commands (name + desc)"""
return '\n'.join(map(
lambda (cname, ccls):
'{0}{1!s:12}{2}'.format(ind, cname,
ccls.__doc__.splitlines()[0]),
self._commands.iteritems()
))
|