Statistics
| Branch: | Tag: | Revision:

root / kamaki / cli / argument / __init__.py @ 2af87afc

History | View | Annotate | Download (14.7 kB)

1
# Copyright 2012-2013 GRNET S.A. All rights reserved.
2
#
3
# Redistribution and use in source and binary forms, with or
4
# without modification, are permitted provided that the following
5
# conditions are met:
6
#
7
#   1. Redistributions of source code must retain the above
8
#     copyright notice, this list of conditions and the following
9
#     disclaimer.
10
#
11
#   2. Redistributions in binary form must reproduce the above
12
#     copyright notice, this list of conditions and the following
13
#     disclaimer in the documentation and/or other materials
14
#     provided with the distribution.
15
#
16
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27
# POSSIBILITY OF SUCH DAMAGE.
28
#
29
# The views and conclusions contained in the software and
30
# documentation are those of the authors and should not be
31
# interpreted as representing official policies, either expressed
32
# or implied, of GRNET S.A.
33

    
34
from kamaki.cli.config import Config
35
from kamaki.cli.errors import CLISyntaxError, raiseCLIError
36
from kamaki.cli.utils import split_input
37

    
38
from datetime import datetime as dtm
39
from time import mktime
40

    
41
from logging import getLogger
42
from argparse import ArgumentParser, ArgumentError
43
from argparse import RawDescriptionHelpFormatter
44

    
45
try:
46
    from progress.bar import ShadyBar as KamakiProgressBar
47
except ImportError:
48
    try:
49
        from progress.bar import Bar as KamakiProgressBar
50
    except ImportError:
51
        pass
52
    # progress not installed - pls, pip install progress
53
    pass
54

    
55
log = getLogger(__name__)
56

    
57

    
58
class Argument(object):
59
    """An argument that can be parsed from command line or otherwise.
60
    This is the top-level Argument class. It is suggested to extent this
61
    class into more specific argument types.
62
    """
63

    
64
    def __init__(self, arity, help=None, parsed_name=None, default=None):
65
        self.arity = int(arity)
66
        self.help = '%s' % help or ''
67

    
68
        assert parsed_name, 'No parsed name for argument %s' % self
69
        self.parsed_name = list(parsed_name) if isinstance(
70
            parsed_name, list) or isinstance(parsed_name, tuple) else (
71
                '%s' % parsed_name).split()
72
        for name in self.parsed_name:
73
            assert name.count(' ') == 0, '%s: Invalid parse name "%s"' % (
74
                self, name)
75
            msg = '%s: Invalid parse name "%s" should start with a "-"' % (
76
                    self, name)
77
            assert name.startswith('-'), msg
78

    
79
        self.default = default or (None if self.arity else False)
80

    
81
    @property
82
    def value(self):
83
        return getattr(self, '_value', self.default)
84

    
85
    @value.setter
86
    def value(self, newvalue):
87
        self._value = newvalue
88

    
89
    def update_parser(self, parser, name):
90
        """Update argument parser with self info"""
91
        action = 'append' if self.arity < 0 else (
92
            'store' if self.arity else 'store_true')
93
        parser.add_argument(
94
            *self.parsed_name,
95
            dest=name, action=action, default=self.default, help=self.help)
96

    
97

    
98
class ConfigArgument(Argument):
99
    """Manage a kamaki configuration (file)"""
100

    
101
    def __init__(self, help, parsed_name=('-c', '--config')):
102
        super(ConfigArgument, self).__init__(1, help, parsed_name, None)
103
        self.file_path = None
104

    
105
    @property
106
    def value(self):
107
        return super(ConfigArgument, self).value
108

    
109
    @value.setter
110
    def value(self, config_file):
111
        if config_file:
112
            self._value = Config(config_file)
113
            self.file_path = config_file
114
        elif self.file_path:
115
            self._value = Config(self.file_path)
116
        else:
117
            self._value = Config()
118

    
119
    def get(self, group, term):
120
        """Get a configuration setting from the Config object"""
121
        return self.value.get(group, term)
122

    
123
    @property
124
    def groups(self):
125
        suffix = '_cli'
126
        slen = len(suffix)
127
        return [term[:-slen] for term in self.value.keys('global') if (
128
            term.endswith(suffix))]
129

    
130
    @property
131
    def cli_specs(self):
132
        suffix = '_cli'
133
        slen = len(suffix)
134
        return [(k[:-slen], v) for k, v in self.value.items('global') if (
135
            k.endswith(suffix))]
136

    
137
    def get_global(self, option):
138
        return self.value.get_global(option)
139

    
140
    def get_cloud(self, cloud, option):
141
        return self.value.get_cloud(cloud, option)
142

    
143

    
144
_config_arg = ConfigArgument('Path to config file')
145

    
146

    
147
class RuntimeConfigArgument(Argument):
148
    """Set a run-time setting option (not persistent)"""
149

    
150
    def __init__(self, config_arg, help='', parsed_name=None, default=None):
151
        super(self.__class__, self).__init__(1, help, parsed_name, default)
152
        self._config_arg = config_arg
153

    
154
    @property
155
    def value(self):
156
        return super(RuntimeConfigArgument, self).value
157

    
158
    @value.setter
159
    def value(self, options):
160
        if options == self.default:
161
            return
162
        if not isinstance(options, list):
163
            options = ['%s' % options]
164
        for option in options:
165
            keypath, sep, val = option.partition('=')
166
            if not sep:
167
                raiseCLIError(
168
                    CLISyntaxError('Argument Syntax Error '),
169
                    details=[
170
                        '%s is missing a "="',
171
                        ' (usage: -o section.key=val)' % option])
172
            section, sep, key = keypath.partition('.')
173
        if not sep:
174
            key = section
175
            section = 'global'
176
        self._config_arg.value.override(
177
            section.strip(),
178
            key.strip(),
179
            val.strip())
180

    
181

    
182
class FlagArgument(Argument):
183
    """
184
    :value: true if set, false otherwise
185
    """
186

    
187
    def __init__(self, help='', parsed_name=None, default=False):
188
        super(FlagArgument, self).__init__(0, help, parsed_name, default)
189

    
190

    
191
class ValueArgument(Argument):
192
    """
193
    :value type: string
194
    :value returns: given value or default
195
    """
196

    
197
    def __init__(self, help='', parsed_name=None, default=None):
198
        super(ValueArgument, self).__init__(1, help, parsed_name, default)
199

    
200

    
201
class CommaSeparatedListArgument(ValueArgument):
202
    """
203
    :value type: string
204
    :value returns: list of the comma separated values
205
    """
206

    
207
    @property
208
    def value(self):
209
        return self._value or list()
210

    
211
    @value.setter
212
    def value(self, newvalue):
213
        self._value = newvalue.split(',') if newvalue else list()
214

    
215

    
216
class IntArgument(ValueArgument):
217

    
218
    @property
219
    def value(self):
220
        """integer (type checking)"""
221
        return getattr(self, '_value', self.default)
222

    
223
    @value.setter
224
    def value(self, newvalue):
225
        try:
226
            self._value = self.default if (
227
                newvalue == self.default) else int(newvalue)
228
        except ValueError:
229
            raiseCLIError(CLISyntaxError(
230
                'IntArgument Error',
231
                details=['Value %s not an int' % newvalue]))
232

    
233

    
234
class DateArgument(ValueArgument):
235

    
236
    DATE_FORMAT = '%a %b %d %H:%M:%S %Y'
237

    
238
    INPUT_FORMATS = [DATE_FORMAT, '%d-%m-%Y', '%H:%M:%S %d-%m-%Y']
239

    
240
    @property
241
    def timestamp(self):
242
        v = getattr(self, '_value', self.default)
243
        return mktime(v.timetuple()) if v else None
244

    
245
    @property
246
    def formated(self):
247
        v = getattr(self, '_value', self.default)
248
        return v.strftime(self.DATE_FORMAT) if v else None
249

    
250
    @property
251
    def value(self):
252
        return self.timestamp
253

    
254
    @value.setter
255
    def value(self, newvalue):
256
        self._value = self.format_date(newvalue) if newvalue else self.default
257

    
258
    def format_date(self, datestr):
259
        for format in self.INPUT_FORMATS:
260
            try:
261
                t = dtm.strptime(datestr, format)
262
            except ValueError:
263
                continue
264
            return t  # .strftime(self.DATE_FORMAT)
265
        raiseCLIError(None, 'Date Argument Error', details=[
266
            '%s not a valid date' % datestr,
267
            'Correct formats:\n\t%s' % self.INPUT_FORMATS])
268

    
269

    
270
class VersionArgument(FlagArgument):
271
    """A flag argument with that prints current version"""
272

    
273
    @property
274
    def value(self):
275
        """bool"""
276
        return super(self.__class__, self).value
277

    
278
    @value.setter
279
    def value(self, newvalue):
280
        self._value = newvalue
281
        if newvalue:
282
            import kamaki
283
            print('kamaki %s' % kamaki.__version__)
284

    
285

    
286
class KeyValueArgument(Argument):
287
    """A Value Argument that can be repeated
288

289
    :syntax: --<arg> key1=value1 --<arg> key2=value2 ...
290
    """
291

    
292
    def __init__(self, help='', parsed_name=None, default=[]):
293
        super(KeyValueArgument, self).__init__(-1, help, parsed_name, default)
294

    
295
    @property
296
    def value(self):
297
        """
298
        :input: key=value
299
        :output: {'key1':'value1', 'key2':'value2', ...}
300
        """
301
        return super(KeyValueArgument, self).value
302

    
303
    @value.setter
304
    def value(self, keyvalue_pairs):
305
        self._value = {}
306
        for pair in keyvalue_pairs:
307
            key, sep, val = pair.partition('=')
308
            if not sep:
309
                raiseCLIError(
310
                    CLISyntaxError('Argument syntax error '),
311
                    details='%s is missing a "=" (usage: key1=val1 )\n' % pair)
312
            self._value[key.strip()] = val.strip()
313

    
314

    
315
class ProgressBarArgument(FlagArgument):
316
    """Manage a progress bar"""
317

    
318
    def __init__(self, help='', parsed_name='', default=True):
319
        self.suffix = '%(percent)d%%'
320
        super(ProgressBarArgument, self).__init__(help, parsed_name, default)
321
        try:
322
            KamakiProgressBar
323
        except NameError:
324
            log.warning('WARNING: no progress bar functionality')
325

    
326
    def clone(self):
327
        """Get a modifiable copy of this bar"""
328
        newarg = ProgressBarArgument(
329
            self.help,
330
            self.parsed_name,
331
            self.default)
332
        newarg._value = self._value
333
        return newarg
334

    
335
    def get_generator(self, message, message_len=25):
336
        """Get a generator to handle progress of the bar (gen.next())"""
337
        if self.value:
338
            return None
339
        try:
340
            self.bar = KamakiProgressBar()
341
        except NameError:
342
            self.value = None
343
            return self.value
344
        self.bar.message = message.ljust(message_len)
345
        self.bar.suffix = '%(percent)d%% - %(eta)ds'
346
        self.bar.start()
347

    
348
        def progress_gen(n):
349
            for i in self.bar.iter(range(int(n))):
350
                yield
351
            yield
352
        return progress_gen
353

    
354
    def finish(self):
355
        """Stop progress bar, return terminal cursor to user"""
356
        if self.value:
357
            return
358
        mybar = getattr(self, 'bar', None)
359
        if mybar:
360
            mybar.finish()
361

    
362

    
363
_arguments = dict(
364
    config=_config_arg,
365
    cloud=ValueArgument('Chose a cloud to connect to', ('--cloud')),
366
    help=Argument(0, 'Show help message', ('-h', '--help')),
367
    debug=FlagArgument('Include debug output', ('-d', '--debug')),
368
    include=FlagArgument(
369
        'Include raw connection data in the output', ('-i', '--include')),
370
    silent=FlagArgument('Do not output anything', ('-s', '--silent')),
371
    verbose=FlagArgument('More info at response', ('-v', '--verbose')),
372
    version=VersionArgument('Print current version', ('-V', '--version')),
373
    options=RuntimeConfigArgument(
374
        _config_arg, 'Override a config value', ('-o', '--options'))
375
)
376

    
377

    
378
#  Initial command line interface arguments
379

    
380

    
381
class ArgumentParseManager(object):
382
    """Manage (initialize and update) an ArgumentParser object"""
383

    
384
    parser = None
385
    _arguments = {}
386
    _parser_modified = False
387
    _parsed = None
388
    _unparsed = None
389

    
390
    def __init__(self, exe, arguments=None):
391
        """
392
        :param exe: (str) the basic command (e.g. 'kamaki')
393

394
        :param arguments: (dict) if given, overrides the global _argument as
395
            the parsers arguments specification
396
        """
397
        self.parser = ArgumentParser(
398
            add_help=False, formatter_class=RawDescriptionHelpFormatter)
399
        self.syntax = '%s <cmd_group> [<cmd_subbroup> ...] <cmd>' % exe
400
        if arguments:
401
            self.arguments = arguments
402
        else:
403
            global _arguments
404
            self.arguments = _arguments
405
        self.parse()
406

    
407
    @property
408
    def syntax(self):
409
        """The command syntax (useful for help messages, descriptions, etc)"""
410
        return self.parser.prog
411

    
412
    @syntax.setter
413
    def syntax(self, new_syntax):
414
        self.parser.prog = new_syntax
415

    
416
    @property
417
    def arguments(self):
418
        """(dict) arguments the parser should be aware of"""
419
        return self._arguments
420

    
421
    @arguments.setter
422
    def arguments(self, new_arguments):
423
        if new_arguments:
424
            assert isinstance(new_arguments, dict)
425
        self._arguments = new_arguments
426
        self.update_parser()
427

    
428
    @property
429
    def parsed(self):
430
        """(Namespace) parser-matched terms"""
431
        if self._parser_modified:
432
            self.parse()
433
        return self._parsed
434

    
435
    @property
436
    def unparsed(self):
437
        """(list) parser-unmatched terms"""
438
        if self._parser_modified:
439
            self.parse()
440
        return self._unparsed
441

    
442
    def update_parser(self, arguments=None):
443
        """Load argument specifications to parser
444

445
        :param arguments: if not given, update self.arguments instead
446
        """
447
        if not arguments:
448
            arguments = self._arguments
449

    
450
        for name, arg in arguments.items():
451
            try:
452
                arg.update_parser(self.parser, name)
453
                self._parser_modified = True
454
            except ArgumentError:
455
                pass
456

    
457
    def update_arguments(self, new_arguments):
458
        """Add to / update existing arguments
459

460
        :param new_arguments: (dict)
461
        """
462
        if new_arguments:
463
            assert isinstance(new_arguments, dict)
464
            self._arguments.update(new_arguments)
465
            self.update_parser()
466

    
467
    def parse(self, new_args=None):
468
        """Parse user input"""
469
        try:
470
            pkargs = (new_args,) if new_args else ()
471
            self._parsed, unparsed = self.parser.parse_known_args(*pkargs)
472
        except SystemExit:
473
            raiseCLIError(CLISyntaxError('Argument Syntax Error'))
474
        for name, arg in self.arguments.items():
475
            arg.value = getattr(self._parsed, name, arg.default)
476
        self._unparsed = []
477
        for term in unparsed:
478
            self._unparsed += split_input(' \'%s\' ' % term)
479
        self._parser_modified = False