Skip to content

Commit

Permalink
Allow to not escape the command on get_wrapped_command
Browse files Browse the repository at this point in the history
  • Loading branch information
humitos committed May 2, 2019
1 parent 08b9a35 commit e58cb78
Show file tree
Hide file tree
Showing 2 changed files with 31 additions and 20 deletions.
7 changes: 4 additions & 3 deletions readthedocs/doc_builder/backends/sphinx.py
Original file line number Diff line number Diff line change
Expand Up @@ -242,16 +242,17 @@ def venv_sphinx_version(self):
In this case, the output will be ``2.0.0``.
"""


command = [
self.python_env.venv_bin(filename='python'),
'-c'
'"import sphinx;print(sphinx.__version__)"',
'"import sphinx; print(sphinx.__version__)"',
]

cmd_ret = self.run(
*command,
cwd=self.project.checkout_path(self.version.slug),
bin_path=self.python_env.venv_bin(),
cwd=self.project.checkout_path(self.version.slug),
escape_command=False,
)
return cmd_ret.output

Expand Down
44 changes: 27 additions & 17 deletions readthedocs/doc_builder/environments.py
Original file line number Diff line number Diff line change
Expand Up @@ -296,14 +296,20 @@ class DockerBuildCommand(BuildCommand):
Build command to execute in docker container
"""

def run(self):
def __init__(self, *args, escape_command=True, **kwargs):
"""
Execute command in existing Docker container.
Override default to extend behavior.
:param cmd_input: input to pass to command in STDIN
:type cmd_input: str
:param combine_output: combine STDERR into STDOUT
:param escape_command: whether escape special chars the command before
executing it in the container. This should only be disabled on
trusted or internal commands.
:type escape_command: bool
"""
self.escape_command = escape_command
super(DockerBuildCommand, self).__init__(*args, **kwargs)

def run(self):
"""Execute command in existing Docker container."""
log.info(
"Running in container %s: '%s' [%s]",
self.build_env.container_id,
Expand All @@ -316,7 +322,7 @@ def run(self):
try:
exec_cmd = client.exec_create(
container=self.build_env.container_id,
cmd=self.get_wrapped_command(),
cmd=self.get_wrapped_command(self.escape_command),
stdout=True,
stderr=True,
)
Expand Down Expand Up @@ -352,13 +358,15 @@ def run(self):

def get_wrapped_command(self):
"""
Escape special bash characters in command to wrap in shell.
Wrap command in a shell and optionally escape special bash characters.
In order to set the current working path inside a docker container, we
need to wrap the command in a shell call manually. Some characters will
be interpreted as shell characters without escaping, such as: ``pip
install requests<0.8``. This escapes a good majority of those
characters.
need to wrap the command in a shell call manually.
Some characters will be interpreted as shell characters without
escaping, such as: ``pip install requests<0.8``. When passing
``escape_command=True`` in the init method this escapes a good majority
of those characters.
"""
bash_escape_re = re.compile(
r"([\t\ \!\"\#\$\&\'\(\)\*\:\;\<\>\?\@"
Expand All @@ -367,16 +375,18 @@ def get_wrapped_command(self):
prefix = ''
if self.bin_path:
prefix += 'PATH={}:$PATH '.format(self.bin_path)

command = (
' '.join([
bash_escape_re.sub(r'\\\1', part) if self.escape_command else part
for part in self.command
])
)
return (
"/bin/sh -c 'cd {cwd} && {prefix}{cmd}'".format(
cwd=self.cwd,
prefix=prefix,
cmd=(
' '.join([
bash_escape_re.sub(r'\\\1', part)
for part in self.command
])
),
cmd=command,
)
)

Expand Down

0 comments on commit e58cb78

Please sign in to comment.