291
276
win32utils.set_file_attr_hidden(path)
295
class TestUnicodeShlex(tests.TestCase):
297
def assertAsTokens(self, expected, line):
298
s = win32utils.UnicodeShlex(line)
299
self.assertEqual(expected, list(s))
301
def test_simple(self):
302
self.assertAsTokens([(False, u'foo'), (False, u'bar'), (False, u'baz')],
305
def test_ignore_multiple_spaces(self):
306
self.assertAsTokens([(False, u'foo'), (False, u'bar')], u'foo bar')
308
def test_ignore_leading_space(self):
309
self.assertAsTokens([(False, u'foo'), (False, u'bar')], u' foo bar')
311
def test_ignore_trailing_space(self):
312
self.assertAsTokens([(False, u'foo'), (False, u'bar')], u'foo bar ')
314
def test_posix_quotations(self):
315
self.assertAsTokens([(True, u'foo bar')], u'"foo bar"')
316
self.assertAsTokens([(False, u"'fo''o"), (False, u"b''ar'")],
318
self.assertAsTokens([(True, u'foo bar')], u'"fo""o b""ar"')
319
self.assertAsTokens([(True, u"fo'o"), (True, u"b'ar")],
322
def test_nested_quotations(self):
323
self.assertAsTokens([(True, u'foo"" bar')], u"\"foo\\\"\\\" bar\"")
324
self.assertAsTokens([(True, u'foo\'\' bar')], u"\"foo'' bar\"")
326
def test_empty_result(self):
327
self.assertAsTokens([], u'')
328
self.assertAsTokens([], u' ')
330
def test_quoted_empty(self):
331
self.assertAsTokens([(True, '')], u'""')
332
self.assertAsTokens([(False, u"''")], u"''")
334
def test_unicode_chars(self):
335
self.assertAsTokens([(False, u'f\xb5\xee'), (False, u'\u1234\u3456')],
336
u'f\xb5\xee \u1234\u3456')
338
def test_newline_in_quoted_section(self):
339
self.assertAsTokens([(True, u'foo\nbar\nbaz\n')], u'"foo\nbar\nbaz\n"')
341
def test_escape_chars(self):
342
self.assertAsTokens([(False, u'foo\\bar')], u'foo\\bar')
344
def test_escape_quote(self):
345
self.assertAsTokens([(True, u'foo"bar')], u'"foo\\"bar"')
347
def test_double_escape(self):
348
self.assertAsTokens([(True, u'foo\\\\bar')], u'"foo\\\\bar"')
349
self.assertAsTokens([(False, u'foo\\\\bar')], u"foo\\\\bar")
351
def test_n_backslashes_handling(self):
352
# https://bugs.launchpad.net/bzr/+bug/528944
353
# actually we care about the doubled backslashes when they're
354
# represents UNC paths.
355
# But in fact there is too much weird corner cases
356
# (see https://bugs.launchpad.net/tortoisebzr/+bug/569050)
357
# so to reproduce every bit of windows command-line handling
358
# could be not worth of efforts?
359
self.requireFeature(BackslashDirSeparatorFeature)
360
self.assertAsTokens([(True, r'\\host\path')], r'"\\host\path"')
361
self.assertAsTokens([(False, r'\\host\path')], r'\\host\path')
362
# handling of " after the 2n and 2n+1 backslashes
363
# inside and outside the quoted string
364
self.assertAsTokens([(True, r'\\'), (False, r'*.py')], r'"\\\\" *.py')
365
self.assertAsTokens([(True, r'\\" *.py')], r'"\\\\\" *.py"')
366
self.assertAsTokens([(True, r'\\ *.py')], r'\\\\" *.py"')
367
self.assertAsTokens([(False, r'\\"'), (False, r'*.py')],
369
self.assertAsTokens([(True, u'\\\\')], u'"\\\\')
372
279
class Test_CommandLineToArgv(tests.TestCaseInTempDir):
374
def assertCommandLine(self, expected, line):
281
def assertCommandLine(self, expected, line, argv=None,
282
single_quotes_allowed=False):
375
283
# Strictly speaking we should respect parameter order versus glob
376
284
# expansions, but it's not really worth the effort here
377
self.assertEqual(expected,
378
sorted(win32utils._command_line_to_argv(line)))
287
argv = win32utils._command_line_to_argv(line, argv,
288
single_quotes_allowed=single_quotes_allowed)
289
self.assertEqual(expected, sorted(argv))
380
291
def test_glob_paths(self):
381
292
self.build_tree(['a/', 'a/b.c', 'a/c.c', 'a/c.h'])
391
302
self.build_tree(['a/', 'a/b.c', 'a/c.c', 'a/c.h'])
392
303
self.assertCommandLine([u'a/*.c'], '"a/*.c"')
393
304
self.assertCommandLine([u"'a/*.c'"], "'a/*.c'")
305
self.assertCommandLine([u'a/*.c'], "'a/*.c'",
306
single_quotes_allowed=True)
395
308
def test_slashes_changed(self):
396
309
# Quoting doesn't change the supplied args
397
310
self.assertCommandLine([u'a\\*.c'], '"a\\*.c"')
311
self.assertCommandLine([u'a\\*.c'], "'a\\*.c'",
312
single_quotes_allowed=True)
398
313
# Expands the glob, but nothing matches, swaps slashes
399
314
self.assertCommandLine([u'a/*.c'], 'a\\*.c')
400
315
self.assertCommandLine([u'a/?.c'], 'a\\?.c')
401
316
# No glob, doesn't touch slashes
402
317
self.assertCommandLine([u'a\\foo.c'], 'a\\foo.c')
404
def test_no_single_quote_supported(self):
319
def test_single_quote_support(self):
405
320
self.assertCommandLine(["add", "let's-do-it.txt"],
406
"add let's-do-it.txt")
321
"add let's-do-it.txt",
322
["add", "let's-do-it.txt"])
323
self.expectFailure("Using single quotes breaks trimming from argv",
324
self.assertCommandLine, ["add", "lets do it.txt"],
325
"add 'lets do it.txt'", ["add", "'lets", "do", "it.txt'"],
326
single_quotes_allowed=True)
408
328
def test_case_insensitive_globs(self):
409
self.requireFeature(tests.CaseInsCasePresFilenameFeature)
329
if os.path.normcase("AbC") == "AbC":
330
self.skip("Test requires case insensitive globbing function")
410
331
self.build_tree(['a/', 'a/b.c', 'a/c.c', 'a/c.h'])
411
332
self.assertCommandLine([u'A/b.c'], 'A/B*')
413
334
def test_backslashes(self):
414
self.requireFeature(BackslashDirSeparatorFeature)
335
self.requireFeature(backslashdir_feature)
415
336
self.build_tree(['a/', 'a/b.c', 'a/c.c', 'a/c.h'])
416
337
self.assertCommandLine([u'a/b.c'], 'a\\b*')
339
def test_with_pdb(self):
340
"""Check stripping Python arguments before bzr script per lp:587868"""
341
self.assertCommandLine([u"rocks"], "-m pdb rocks", ["rocks"])
342
self.build_tree(['d/', 'd/f1', 'd/f2'])
343
self.assertCommandLine([u"rm", u"x*"], "-m pdb rm x*", ["rm", u"x*"])
344
self.assertCommandLine([u"add", u"d/f1", u"d/f2"], "-m pdb add d/*",