parent
06bc4807a8
commit
6683010baa
@ -0,0 +1,93 @@
|
||||
# HG changeset patch
|
||||
# User Clark Boylan <clark.boylan@gmail.com>
|
||||
# Date 1391830704 28800
|
||||
# Node ID 62fe57a8fd3f8f44be8957e59846387d2f505227
|
||||
# Parent b0360a54ab368ef428c7f83601ba6b64f6fec64f
|
||||
Fix command expansion and parsing.
|
||||
|
||||
Tox testenv commands are parsed to expand variable substitutions and
|
||||
construct the argv list that will be passed to exec. Prior to this
|
||||
commit this parsing ate quotes surrounding variables and treated
|
||||
multiword variables as single argv items. Neither behavior was correct.
|
||||
To fix this create the expanded command before handing it off to shlex
|
||||
to do the tokenization of the argv list. Doing the parsing in this
|
||||
order ensures it is correct.
|
||||
|
||||
diff --git a/tests/test_config.py b/tests/test_config.py
|
||||
--- a/tests/test_config.py
|
||||
+++ b/tests/test_config.py
|
||||
@@ -278,7 +278,7 @@
|
||||
# "reader.getargvlist('section', 'key1')")
|
||||
assert reader.getargvlist('section', 'key1') == []
|
||||
x = reader.getargvlist("section", "key2")
|
||||
- assert x == [["cmd1", "with space", "grr"],
|
||||
+ assert x == [["cmd1", "with", "space", "grr"],
|
||||
["cmd2", "grr"]]
|
||||
|
||||
def test_argvlist_windows_escaping(self, tmpdir, newconfig):
|
||||
@@ -304,7 +304,7 @@
|
||||
# "reader.getargvlist('section', 'key1')")
|
||||
assert reader.getargvlist('section', 'key1') == []
|
||||
x = reader.getargvlist("section", "key2")
|
||||
- assert x == [["cmd1", "with space", "grr"]]
|
||||
+ assert x == [["cmd1", "with", "space", "grr"]]
|
||||
|
||||
|
||||
def test_argvlist_quoting_in_command(self, tmpdir, newconfig):
|
||||
diff --git a/tox/_config.py b/tox/_config.py
|
||||
--- a/tox/_config.py
|
||||
+++ b/tox/_config.py
|
||||
@@ -527,30 +527,35 @@
|
||||
def _processcommand(self, command):
|
||||
posargs = getattr(self, "posargs", None)
|
||||
|
||||
- # special treat posargs which might contain multiple arguments
|
||||
- # in their defaults
|
||||
+ # Iterate through each word of the command substituting as
|
||||
+ # appropriate to construct the new command string. This
|
||||
+ # string is then broken up into exec argv components using
|
||||
+ # shlex.
|
||||
newcommand = ""
|
||||
for word in CommandParser(command).words():
|
||||
- if word.startswith("{posargs:") and word.endswith("}"):
|
||||
+ if word == "{posargs}" or word == "[]":
|
||||
if posargs:
|
||||
- word = "{posargs}"
|
||||
+ newcommand += " ".join(posargs)
|
||||
+ continue
|
||||
+ elif word.startswith("{posargs:") and word.endswith("}"):
|
||||
+ if posargs:
|
||||
+ newcommand += " ".join(posargs)
|
||||
+ continue
|
||||
else:
|
||||
word = word[9:-1]
|
||||
- newcommand += word
|
||||
+ new_arg = ""
|
||||
+ new_word = self._replace(word)
|
||||
+ new_word = self._replace(new_word)
|
||||
+ new_arg += new_word
|
||||
+ newcommand += new_arg
|
||||
|
||||
- # now we can properly parse the command
|
||||
- argv = []
|
||||
- for arg in shlex.split(newcommand):
|
||||
- if arg in ('[]', "{posargs}"):
|
||||
- if posargs:
|
||||
- argv.extend(posargs)
|
||||
- continue
|
||||
- new_arg = ""
|
||||
- for word in CommandParser(arg).words():
|
||||
- new_word = self._replace(word)
|
||||
- new_word = self._replace(new_word)
|
||||
- new_arg += new_word
|
||||
- argv.append(new_arg)
|
||||
+ # Construct shlex object that will not escape any values,
|
||||
+ # use all values as is in argv.
|
||||
+ shlexer = shlex.shlex(newcommand, posix=True)
|
||||
+ shlexer.whitespace_split = True
|
||||
+ shlexer.escape = ''
|
||||
+ shlexer.commenters = ''
|
||||
+ argv = list(shlexer)
|
||||
return argv
|
||||
|
||||
def getargv(self, section, name, default=None, replace=True):
|
Loading…
Reference in new issue