41
43
each revision specifier supplied.
43
45
>>> _parse_revision_str('234')
44
[<RevisionSpec_revno 234>]
46
[<RevisionSpec_dwim 234>]
45
47
>>> _parse_revision_str('234..567')
46
[<RevisionSpec_revno 234>, <RevisionSpec_revno 567>]
48
[<RevisionSpec_dwim 234>, <RevisionSpec_dwim 567>]
47
49
>>> _parse_revision_str('..')
48
50
[<RevisionSpec None>, <RevisionSpec None>]
49
51
>>> _parse_revision_str('..234')
50
[<RevisionSpec None>, <RevisionSpec_revno 234>]
52
[<RevisionSpec None>, <RevisionSpec_dwim 234>]
51
53
>>> _parse_revision_str('234..')
52
[<RevisionSpec_revno 234>, <RevisionSpec None>]
54
[<RevisionSpec_dwim 234>, <RevisionSpec None>]
53
55
>>> _parse_revision_str('234..456..789') # Maybe this should be an error
54
[<RevisionSpec_revno 234>, <RevisionSpec_revno 456>, <RevisionSpec_revno 789>]
56
[<RevisionSpec_dwim 234>, <RevisionSpec_dwim 456>, <RevisionSpec_dwim 789>]
55
57
>>> _parse_revision_str('234....789') #Error ?
56
[<RevisionSpec_revno 234>, <RevisionSpec None>, <RevisionSpec_revno 789>]
58
[<RevisionSpec_dwim 234>, <RevisionSpec None>, <RevisionSpec_dwim 789>]
57
59
>>> _parse_revision_str('revid:test@other.com-234234')
58
60
[<RevisionSpec_revid revid:test@other.com-234234>]
59
61
>>> _parse_revision_str('revid:test@other.com-234234..revid:test@other.com-234235')
60
62
[<RevisionSpec_revid revid:test@other.com-234234>, <RevisionSpec_revid revid:test@other.com-234235>]
61
63
>>> _parse_revision_str('revid:test@other.com-234234..23')
62
[<RevisionSpec_revid revid:test@other.com-234234>, <RevisionSpec_revno 23>]
64
[<RevisionSpec_revid revid:test@other.com-234234>, <RevisionSpec_dwim 23>]
63
65
>>> _parse_revision_str('date:2005-04-12')
64
66
[<RevisionSpec_date date:2005-04-12>]
65
67
>>> _parse_revision_str('date:2005-04-12 12:24:33')
69
71
>>> _parse_revision_str('date:2005-04-12,12:24:33')
70
72
[<RevisionSpec_date date:2005-04-12,12:24:33>]
71
73
>>> _parse_revision_str('-5..23')
72
[<RevisionSpec_revno -5>, <RevisionSpec_revno 23>]
74
[<RevisionSpec_dwim -5>, <RevisionSpec_dwim 23>]
73
75
>>> _parse_revision_str('-5')
74
[<RevisionSpec_revno -5>]
76
[<RevisionSpec_dwim -5>]
75
77
>>> _parse_revision_str('123a')
76
Traceback (most recent call last):
78
NoSuchRevisionSpec: No namespace registered for string: '123a'
78
[<RevisionSpec_dwim 123a>]
79
79
>>> _parse_revision_str('abc')
80
Traceback (most recent call last):
82
NoSuchRevisionSpec: No namespace registered for string: 'abc'
80
[<RevisionSpec_dwim abc>]
83
81
>>> _parse_revision_str('branch:../branch2')
84
82
[<RevisionSpec_branch branch:../branch2>]
85
83
>>> _parse_revision_str('branch:../../branch2')
86
84
[<RevisionSpec_branch branch:../../branch2>]
87
85
>>> _parse_revision_str('branch:../../branch2..23')
88
[<RevisionSpec_branch branch:../../branch2>, <RevisionSpec_revno 23>]
86
[<RevisionSpec_branch branch:../../branch2>, <RevisionSpec_dwim 23>]
89
87
>>> _parse_revision_str('branch:..\\\\branch2')
90
88
[<RevisionSpec_branch branch:..\\branch2>]
91
89
>>> _parse_revision_str('branch:..\\\\..\\\\branch2..23')
92
[<RevisionSpec_branch branch:..\\..\\branch2>, <RevisionSpec_revno 23>]
90
[<RevisionSpec_branch branch:..\\..\\branch2>, <RevisionSpec_dwim 23>]
94
92
# TODO: Maybe move this into revisionspec.py
208
211
option_strings = ['--%s' % self.name]
209
212
if short_name is not None:
210
213
option_strings.append('-%s' % short_name)
215
help = optparse.SUPPRESS_HELP
211
218
optargfn = self.type
212
219
if optargfn is None:
213
parser.add_option(action='callback',
214
callback=self._optparse_bool_callback,
220
parser.add_option(action='callback',
221
callback=self._optparse_bool_callback,
215
222
callback_args=(True,),
218
225
negation_strings = ['--%s' % self.get_negation_name()]
219
parser.add_option(action='callback',
220
callback=self._optparse_bool_callback,
226
parser.add_option(action='callback',
227
callback=self._optparse_bool_callback,
221
228
callback_args=(False,),
222
229
help=optparse.SUPPRESS_HELP, *negation_strings)
224
parser.add_option(action='callback',
225
callback=self._optparse_callback,
231
parser.add_option(action='callback',
232
callback=self._optparse_callback,
226
233
type='string', metavar=self.argname.upper(),
228
default=OptionParser.DEFAULT_VALUE,
235
default=OptionParser.DEFAULT_VALUE,
231
238
def _optparse_bool_callback(self, option, opt_str, value, parser, bool_v):
321
329
'--knit' can be used interchangeably.
322
330
:param enum_switch: If true, a switch is provided with the option name,
323
331
which takes a value.
332
:param lazy_registry: A tuple of (module name, attribute name) for a
333
registry to be lazily loaded.
334
:param short_name: The short name for the enum switch, if any
335
:param short_value_switches: A dict mapping values to short names
325
Option.__init__(self, name, help, type=self.convert)
326
self.registry = registry
337
Option.__init__(self, name, help, type=self.convert,
338
short_name=short_name)
339
self._registry = registry
341
if lazy_registry is None:
342
raise AssertionError(
343
'One of registry or lazy_registry must be given.')
344
self._lazy_registry = _mod_registry._LazyObjectGetter(
346
if registry is not None and lazy_registry is not None:
347
raise AssertionError(
348
'registry and lazy_registry are mutually exclusive')
328
350
self.converter = converter
329
351
self.value_switches = value_switches
330
352
self.enum_switch = enum_switch
353
self.short_value_switches = short_value_switches
331
354
self.title = title
332
355
if self.title is None:
333
356
self.title = name
360
if self._registry is None:
361
self._registry = self._lazy_registry.get_obj()
362
return self._registry
336
365
def from_kwargs(name_, help=None, title=None, value_switches=False,
337
366
enum_switch=True, **kwargs):
340
369
name, help, value_switches and enum_switch are passed to the
341
370
RegistryOption constructor. Any other keyword arguments are treated
342
as values for the option, and they value is treated as the help.
371
as values for the option, and their value is treated as the help.
344
reg = registry.Registry()
345
for name, switch_help in kwargs.iteritems():
373
reg = _mod_registry.Registry()
374
for name, switch_help in sorted(kwargs.items()):
346
375
name = name.replace('_', '-')
347
376
reg.register(name, name, help=switch_help)
377
if not value_switches:
378
help = help + ' "' + name + '": ' + switch_help
379
if not help.endswith("."):
348
381
return RegistryOption(name_, help, reg, title=title,
349
382
value_switches=value_switches, enum_switch=enum_switch)
424
477
Option.STD_OPTIONS[name] = Option(name, **kwargs)
425
478
Option.OPTIONS[name] = Option.STD_OPTIONS[name]
480
def _standard_list_option(name, **kwargs):
481
"""Register a standard option."""
482
# All standard options are implicitly 'global' ones
483
Option.STD_OPTIONS[name] = ListOption(name, **kwargs)
484
Option.OPTIONS[name] = Option.STD_OPTIONS[name]
428
487
def _global_option(name, **kwargs):
429
488
"""Register a global option."""
430
489
Option.OPTIONS[name] = Option(name, **kwargs)
433
def _global_registry_option(name, help, registry, **kwargs):
492
def _global_registry_option(name, help, registry=None, **kwargs):
434
493
Option.OPTIONS[name] = RegistryOption(name, help, registry, **kwargs)
437
class MergeTypeRegistry(registry.Registry):
442
496
# This is the verbosity level detected during command line parsing.
443
497
# Note that the final value is dependent on the order in which the
444
498
# various flags (verbose, quiet, no-verbose, no-quiet) are given.
467
521
_verbosity_level = -1
470
_merge_type_registry = MergeTypeRegistry()
471
_merge_type_registry.register_lazy('merge3', 'bzrlib.merge', 'Merge3Merger',
472
"Native diff3-style merge")
473
_merge_type_registry.register_lazy('diff3', 'bzrlib.merge', 'Diff3Merger',
474
"Merge using external diff3")
475
_merge_type_registry.register_lazy('weave', 'bzrlib.merge', 'WeaveMerger',
478
524
# Declare the standard options
479
525
_standard_option('help', short_name='h',
480
526
help='Show help message.')
527
_standard_option('quiet', short_name='q',
528
help="Only display errors and warnings.",
529
custom_callback=_verbosity_level_callback)
530
_standard_option('usage',
531
help='Show usage message and options.')
481
532
_standard_option('verbose', short_name='v',
482
533
help='Display more information.',
483
534
custom_callback=_verbosity_level_callback)
484
_standard_option('quiet', short_name='q',
485
help="Only display errors and warnings.",
486
custom_callback=_verbosity_level_callback)
488
536
# Declare commonly used options
489
_global_option('all')
490
_global_option('overwrite', help='Ignore differences between branches and '
491
'overwrite unconditionally.')
492
_global_option('basis', type=str)
493
_global_option('bound')
494
_global_option('diff-options', type=str)
537
_global_option('change',
538
type=_parse_change_str,
540
param_name='revision',
541
help='Select changes introduced by the specified revision. See also "help revisionspec".')
542
_global_option('directory', short_name='d', type=unicode,
543
help='Branch to operate on, instead of working directory.')
495
544
_global_option('file', type=unicode, short_name='F')
496
_global_option('force')
497
_global_option('format', type=unicode)
498
_global_option('forward')
545
_global_registry_option('log-format', "Use specified log format.",
546
lazy_registry=('bzrlib.log', 'log_formatter_registry'),
547
value_switches=True, title='Log format',
548
short_value_switches={'short': 'S'})
549
_global_registry_option('merge-type', 'Select a particular merge algorithm.',
550
lazy_registry=('bzrlib.merge', 'merge_type_registry'),
551
value_switches=True, title='Merge algorithm')
499
552
_global_option('message', type=unicode,
501
554
help='Message string.')
502
_global_option('no-recurse')
503
_global_option('profile',
504
help='Show performance profiling information.')
555
_global_option('null', short_name='0',
556
help='Use an ASCII NUL (\\0) separator rather than '
558
_global_option('overwrite', help='Ignore differences between branches and '
559
'overwrite unconditionally.')
560
_global_option('remember', help='Remember the specified location as a'
562
_global_option('reprocess', help='Reprocess to reduce spurious conflicts.')
505
563
_global_option('revision',
506
564
type=_parse_revision_str,
508
566
help='See "help revisionspec" for details.')
509
_global_option('change',
510
type=_parse_change_str,
512
param_name='revision',
513
help='Select changes introduced by the specified revision. See also "help revisionspec".')
514
567
_global_option('show-ids',
515
568
help='Show internal object ids.')
516
_global_option('timezone',
569
_global_option('timezone',
518
help='display timezone as local, original, or utc')
519
_global_option('unbound')
520
_global_option('version')
521
_global_option('email')
522
_global_option('update')
523
_global_registry_option('log-format', "Use specified log format.",
524
log.log_formatter_registry, value_switches=True,
526
_global_option('long', help='Use detailed log format. Same as --log-format long',
528
_global_option('short', help='Use moderately short log format. Same as --log-format short')
529
_global_option('line', help='Use log format with one line per revision. Same as --log-format line')
530
_global_option('root', type=str)
531
_global_option('no-backup')
532
_global_registry_option('merge-type', 'Select a particular merge algorithm.',
533
_merge_type_registry, value_switches=True,
534
title='Merge algorithm')
535
_global_option('pattern', type=str)
536
_global_option('remember', help='Remember the specified location as a'
538
_global_option('reprocess', help='Reprocess to reduce spurious conflicts.')
539
_global_option('kind', type=str)
540
_global_option('dry-run',
541
help="Show what would be done, but don't actually do anything.")
542
_global_option('name-from-revision', help='The path name in the old tree.')
571
help='Display timezone as local, original, or utc.')
573
diff_writer_registry = _mod_registry.Registry()
574
diff_writer_registry.register('plain', lambda x: x, 'Plaintext diff output.')
575
diff_writer_registry.default_key = 'plain'