214
235
('two', None, None, 'two help'),
238
def test_option_callback_bool(self):
239
"Test booleans get True and False passed correctly to a callback."""
241
def cb(option, name, value, parser):
242
cb_calls.append((option,name,value,parser))
243
options = [option.Option('hello', custom_callback=cb)]
244
opts, args = self.parse(options, ['--hello', '--no-hello'])
245
self.assertEqual(2, len(cb_calls))
246
opt,name,value,parser = cb_calls[0]
247
self.assertEqual('hello', name)
248
self.assertTrue(value)
249
opt,name,value,parser = cb_calls[1]
250
self.assertEqual('hello', name)
251
self.assertFalse(value)
253
def test_option_callback_str(self):
254
"""Test callbacks work for string options both long and short."""
256
def cb(option, name, value, parser):
257
cb_calls.append((option,name,value,parser))
258
options = [option.Option('hello', type=str, custom_callback=cb,
260
opts, args = self.parse(options, ['--hello', 'world', '-h', 'mars'])
261
self.assertEqual(2, len(cb_calls))
262
opt,name,value,parser = cb_calls[0]
263
self.assertEqual('hello', name)
264
self.assertEqual('world', value)
265
opt,name,value,parser = cb_calls[1]
266
self.assertEqual('hello', name)
267
self.assertEqual('mars', value)
218
270
class TestListOptions(TestCase):
219
271
"""Tests for ListOption, used to specify lists on the command-line."""
250
308
options, ['--hello=a', '--hello=b', '--hello=-', '--hello=c'])
251
309
self.assertEqual(['c'], opts.hello)
311
def test_option_callback_list(self):
312
"""Test callbacks work for list options."""
314
def cb(option, name, value, parser):
315
# Note that the value is a reference so copy to keep it
316
cb_calls.append((option,name,value[:],parser))
317
options = [option.ListOption('hello', type=str, custom_callback=cb)]
318
opts, args = self.parse(options, ['--hello=world', '--hello=mars',
320
self.assertEqual(3, len(cb_calls))
321
opt,name,value,parser = cb_calls[0]
322
self.assertEqual('hello', name)
323
self.assertEqual(['world'], value)
324
opt,name,value,parser = cb_calls[1]
325
self.assertEqual('hello', name)
326
self.assertEqual(['world', 'mars'], value)
327
opt,name,value,parser = cb_calls[2]
328
self.assertEqual('hello', name)
329
self.assertEqual([], value)
331
def test_list_option_param_name(self):
332
"""Test list options can have their param_name set."""
333
options = [option.ListOption('hello', type=str, param_name='greeting')]
334
opts, args = self.parse(
335
options, ['--hello=world', '--hello=sailor'])
336
self.assertEqual(['world', 'sailor'], opts.greeting)
254
339
class TestOptionDefinitions(TestCase):
255
340
"""Tests for options in the Bazaar codebase."""
257
342
def get_builtin_command_options(self):
259
for cmd_name, cmd_class in sorted(commands.get_all_cmds()):
344
commands.install_bzr_command_hooks()
345
for cmd_name in sorted(commands.builtin_command_names()):
346
cmd = commands.get_cmd_object(cmd_name)
261
347
for opt_name, opt in sorted(cmd.options().items()):
262
348
g.append((cmd_name, opt))
265
def test_global_options_used(self):
266
# In the distant memory, options could only be declared globally. Now
267
# we prefer to declare them in the command, unless like -r they really
268
# are used very widely with the exact same meaning. So this checks
269
# for any that should be garbage collected.
270
g = dict(option.Option.OPTIONS.items())
273
for cmd_name, cmd_class in sorted(commands.get_all_cmds()):
274
for option_or_name in sorted(cmd_class.takes_options):
275
if not isinstance(option_or_name, basestring):
276
self.assertIsInstance(option_or_name, option.Option)
277
elif not option_or_name in g:
278
msgs.append("apparent reference to undefined "
279
"global option %r from %r"
280
% (option_or_name, cmd_class))
282
used_globals.setdefault(option_or_name, []).append(cmd_name)
283
unused_globals = set(g.keys()) - set(used_globals.keys())
284
# not enforced because there might be plugins that use these globals
285
## for option_name in sorted(unused_globals):
286
## msgs.append("unused global option %r" % option_name)
287
## for option_name, cmds in sorted(used_globals.items()):
288
## if len(cmds) <= 1:
289
## msgs.append("global option %r is only used by %r"
290
## % (option_name, cmds))
292
self.fail("problems with global option definitions:\n"
295
352
def test_option_grammar(self):
297
354
# Option help should be written in sentence form, and have a final
298
# period and be all on a single line, because the display code will
300
option_re = re.compile(r'^[A-Z][^\n]+\.$')
301
for scope, option in self.get_builtin_command_options():
303
msgs.append('%-16s %-16s %s' %
304
((scope or 'GLOBAL'), option.name, 'NO HELP'))
305
elif not option_re.match(option.help):
306
msgs.append('%-16s %-16s %s' %
307
((scope or 'GLOBAL'), option.name, option.help))
355
# period with an optional bracketed suffix. All the text should be on
356
# one line, because the display code will wrap it.
357
option_re = re.compile(r'^[A-Z][^\n]+\.(?: \([^\n]+\))?$')
358
for scope, opt in self.get_builtin_command_options():
359
for name, _, _, helptxt in opt.iter_switches():
361
name = "/".join([opt.name, name])
363
msgs.append('%-16s %-16s %s' %
364
((scope or 'GLOBAL'), name, 'NO HELP'))
365
elif not option_re.match(helptxt):
366
if name.startswith("format/"):
367
# Don't complain about the odd format registry help
369
msgs.append('%-16s %-16s %s' %
370
((scope or 'GLOBAL'), name, helptxt))
309
372
self.fail("The following options don't match the style guide:\n"
310
373
+ '\n'.join(msgs))
376
class TestOptionMisc(TestCase):
312
378
def test_is_hidden(self):
313
registry = bzrdir.BzrDirFormatRegistry()
314
registry.register_metadir('hidden', 'HiddenFormat',
379
registry = controldir.ControlDirFormatRegistry()
380
bzrdir.register_metadir(registry, 'hidden', 'HiddenFormat',
315
381
'hidden help text', hidden=True)
316
registry.register_metadir('visible', 'VisibleFormat',
382
bzrdir.register_metadir(registry, 'visible', 'VisibleFormat',
317
383
'visible help text', hidden=False)
318
384
format = option.RegistryOption('format', '', registry, str)
319
385
self.assertTrue(format.is_hidden('hidden'))
320
386
self.assertFalse(format.is_hidden('visible'))
388
def test_short_name(self):
389
registry = controldir.ControlDirFormatRegistry()
390
opt = option.RegistryOption('format', help='', registry=registry)
391
self.assertEquals(None, opt.short_name())
392
opt = option.RegistryOption('format', short_name='F', help='',
394
self.assertEquals('F', opt.short_name())
396
def test_option_custom_help(self):
397
the_opt = option.Option.OPTIONS['help']
398
orig_help = the_opt.help[:]
399
my_opt = option.custom_help('help', 'suggest lottery numbers')
400
# Confirm that my_opt has my help and the original is unchanged
401
self.assertEqual('suggest lottery numbers', my_opt.help)
402
self.assertEqual(orig_help, the_opt.help)
404
def test_short_value_switches(self):
405
reg = registry.Registry()
406
reg.register('short', 'ShortChoice')
407
reg.register('long', 'LongChoice')
408
ropt = option.RegistryOption('choice', '', reg, value_switches=True,
409
short_value_switches={'short': 's'})
410
opts, args = parse([ropt], ['--short'])
411
self.assertEqual('ShortChoice', opts.choice)
412
opts, args = parse([ropt], ['-s'])
413
self.assertEqual('ShortChoice', opts.choice)
416
class TestVerboseQuietLinkage(TestCase):
418
def check(self, parser, level, args):
419
option._verbosity_level = 0
420
opts, args = parser.parse_args(args)
421
self.assertEqual(level, option._verbosity_level)
423
def test_verbose_quiet_linkage(self):
424
parser = option.get_optparser(option.Option.STD_OPTIONS)
425
self.check(parser, 0, [])
426
self.check(parser, 1, ['-v'])
427
self.check(parser, 2, ['-v', '-v'])
428
self.check(parser, -1, ['-q'])
429
self.check(parser, -2, ['-qq'])
430
self.check(parser, -1, ['-v', '-v', '-q'])
431
self.check(parser, 2, ['-q', '-v', '-v'])
432
self.check(parser, 0, ['--no-verbose'])
433
self.check(parser, 0, ['-v', '-q', '--no-quiet'])