Skip to content

Commit 34884b2

Browse files
gh-81691: Fix handling of multiple "--" (double dashes) in argparse
Only the first one has now been removed, all subsequent ones are now taken literally.
1 parent 81480e6 commit 34884b2

File tree

3 files changed

+70
-8
lines changed

3 files changed

+70
-8
lines changed

Lib/argparse.py

Lines changed: 8 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -1916,12 +1916,14 @@ def _parse_known_args(self, arg_strings, namespace):
19161916
# which has an 'O' if there is an option at an index,
19171917
# an 'A' if there is an argument, or a '-' if there is a '--'
19181918
option_string_indices = {}
1919+
double_dash_index = len(arg_strings)
19191920
arg_string_pattern_parts = []
19201921
arg_strings_iter = iter(arg_strings)
19211922
for i, arg_string in enumerate(arg_strings_iter):
19221923

19231924
# all args after -- are non-options
19241925
if arg_string == '--':
1926+
double_dash_index = i
19251927
arg_string_pattern_parts.append('-')
19261928
for arg_string in arg_strings_iter:
19271929
arg_string_pattern_parts.append('A')
@@ -2070,6 +2072,12 @@ def consume_positionals(start_index):
20702072
# and add the Positional and its args to the list
20712073
for action, arg_count in zip(positionals, arg_counts):
20722074
args = arg_strings[start_index: start_index + arg_count]
2075+
# Strip out the first '--' if it is not in PARSER or REMAINDER arg.
2076+
if (arg_count
2077+
and start_index <= double_dash_index < start_index + arg_count
2078+
and action.nargs not in [PARSER, REMAINDER]):
2079+
assert args.index('--') == double_dash_index - start_index
2080+
args.remove('--')
20732081
start_index += arg_count
20742082
if args and action.deprecated and action.dest not in warned:
20752083
self._warning(_("argument '%(argument_name)s' is deprecated") %
@@ -2470,13 +2478,6 @@ def parse_known_intermixed_args(self, args=None, namespace=None):
24702478
# Value conversion methods
24712479
# ========================
24722480
def _get_values(self, action, arg_strings):
2473-
# for everything but PARSER, REMAINDER args, strip out first '--'
2474-
if not action.option_strings and action.nargs not in [PARSER, REMAINDER]:
2475-
try:
2476-
arg_strings.remove('--')
2477-
except ValueError:
2478-
pass
2479-
24802481
# optional argument produces a default when not present
24812482
if not arg_strings and action.nargs == OPTIONAL:
24822483
if action.option_strings:

Lib/test/test_argparse.py

Lines changed: 59 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -5721,7 +5721,30 @@ def test_zero_or_more_optional(self):
57215721
self.assertEqual(NS(x=[]), args)
57225722

57235723
def test_double_dash(self):
5724-
parser = argparse.ArgumentParser()
5724+
parser = argparse.ArgumentParser(exit_on_error=False)
5725+
parser.add_argument('-f', '--foo')
5726+
parser.add_argument('bar', nargs='*')
5727+
5728+
args = parser.parse_args(['--foo=--'])
5729+
self.assertEqual(NS(foo='--', bar=[]), args)
5730+
self.assertRaisesRegex(argparse.ArgumentError,
5731+
'argument -f/--foo: expected one argument',
5732+
parser.parse_args, ['--foo', '--'])
5733+
args = parser.parse_args(['-f--'])
5734+
self.assertEqual(NS(foo='--', bar=[]), args)
5735+
self.assertRaisesRegex(argparse.ArgumentError,
5736+
'argument -f/--foo: expected one argument',
5737+
parser.parse_args, ['-f', '--'])
5738+
args = parser.parse_args(['--foo', 'a', '--', 'b', 'c'])
5739+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5740+
args = parser.parse_args(['a', 'b', '--foo', 'c'])
5741+
self.assertEqual(NS(foo='c', bar=['a', 'b']), args)
5742+
args = parser.parse_args(['a', '--', 'b', '--foo', 'c'])
5743+
self.assertEqual(NS(foo=None, bar=['a', 'b', '--foo', 'c']), args)
5744+
args = parser.parse_args(['a', '--', 'b', '--', 'c', '--foo', 'd'])
5745+
self.assertEqual(NS(foo=None, bar=['a', 'b', '--', 'c', '--foo', 'd']), args)
5746+
5747+
parser = argparse.ArgumentParser(exit_on_error=False)
57255748
parser.add_argument('-f', '--foo', nargs='*')
57265749
parser.add_argument('bar', nargs='*')
57275750

@@ -5735,6 +5758,41 @@ def test_double_dash(self):
57355758
self.assertEqual(NS(foo=[], bar=[]), args)
57365759
args = parser.parse_args(['--foo', 'a', 'b', '--', 'c', 'd'])
57375760
self.assertEqual(NS(foo=['a', 'b'], bar=['c', 'd']), args)
5761+
args = parser.parse_args(['a', 'b', '--foo', 'c', 'd'])
5762+
self.assertEqual(NS(foo=['c', 'd'], bar=['a', 'b']), args)
5763+
args = parser.parse_args(['a', '--', 'b', '--foo', 'c', 'd'])
5764+
self.assertEqual(NS(foo=None, bar=['a', 'b', '--foo', 'c', 'd']), args)
5765+
args, argv = parser.parse_known_args(['a', 'b', '--foo', 'c', '--', 'd'])
5766+
self.assertEqual(NS(foo=['c'], bar=['a', 'b']), args)
5767+
self.assertEqual(argv, ['--', 'd'])
5768+
5769+
parser = argparse.ArgumentParser(exit_on_error=False)
5770+
parser.add_argument('foo')
5771+
parser.add_argument('bar', nargs='*')
5772+
5773+
args = parser.parse_args(['--', 'a', 'b', 'c'])
5774+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5775+
args = parser.parse_args(['a', '--', 'b', 'c'])
5776+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5777+
args = parser.parse_args(['a', 'b', '--', 'c'])
5778+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5779+
args = parser.parse_args(['a', '--', 'b', '--', 'c'])
5780+
self.assertEqual(NS(foo='a', bar=['b', '--', 'c']), args)
5781+
args = parser.parse_args(['--', '--', 'a', '--', 'b', 'c'])
5782+
self.assertEqual(NS(foo='--', bar=['a', '--', 'b', 'c']), args)
5783+
5784+
parser = argparse.ArgumentParser(exit_on_error=False)
5785+
parser.add_argument('foo')
5786+
parser.add_argument('bar', nargs=argparse.REMAINDER)
5787+
5788+
args = parser.parse_args(['--', 'a', 'b', 'c'])
5789+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5790+
args = parser.parse_args(['a', '--', 'b', 'c'])
5791+
self.assertEqual(NS(foo='a', bar=['b', 'c']), args)
5792+
args = parser.parse_args(['a', 'b', '--', 'c'])
5793+
self.assertEqual(NS(foo='a', bar=['b', '--', 'c']), args)
5794+
args = parser.parse_args(['a', '--', 'b', '--', 'c'])
5795+
self.assertEqual(NS(foo='a', bar=['b', '--', 'c']), args)
57385796

57395797

57405798
# ===========================
Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
Fix handling of multiple ``"--"`` (double dashes) in :mod:`argparse`. Only
2+
the first one has now been removed, all subsequent ones are now taken
3+
literally.

0 commit comments

Comments
 (0)