| # Copyright 2018 Google Inc. All Rights Reserved. | |
| # | |
| # Licensed under the Apache License, Version 2.0 (the "License"); | |
| # you may not use this file except in compliance with the License. | |
| # You may obtain a copy of the License at | |
| # | |
| # http://www.apache.org/licenses/LICENSE-2.0 | |
| # | |
| # Unless required by applicable law or agreed to in writing, software | |
| # distributed under the License is distributed on an "AS IS" BASIS, | |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
| # See the License for the specific language governing permissions and | |
| # limitations under the License. | |
| """Allows to use cmd as a shell.""" | |
| import re | |
| from starboard.tools.toolchain import abstract | |
| def _MaybeJoin(shell, command): | |
| if isinstance(command, basestring): | |
| return command | |
| return shell.Join(command) | |
| class Shell(abstract.Shell): | |
| """Constructs command lines using Cmd syntax.""" | |
| def MaybeQuoteArgument(self, arg): | |
| # Rather than attempting to enumerate the bad shell characters, just | |
| # whitelist common OK ones and quote anything else. | |
| if re.match(r'^[a-zA-Z0-9_=.\\/-]+$', arg): | |
| return arg # No quoting necessary. | |
| return self.QuoteForRspFile(arg) | |
| def QuoteForRspFile(self, arg): | |
| """Quote a command line argument. | |
| Quote the argument so that it appears as one argument when | |
| processed via cmd.exe and parsed by CommandLineToArgvW (as is typical for | |
| Windows programs). | |
| Args: | |
| arg: The argument to quote. | |
| Returns: | |
| The quoted argument. | |
| """ | |
| # See http://goo.gl/cuFbX and http://goo.gl/dhPnp including the comment | |
| # threads. This is actually the quoting rules for CommandLineToArgvW, not | |
| # for the shell, because the shell doesn't do anything in Windows. This | |
| # works more or less because most programs (including the compiler, etc.) | |
| # use that function to handle command line arguments. | |
| # For a literal quote, CommandLineToArgvW requires 2n+1 backslashes | |
| # preceding it, and results in n backslashes + the quote. So we substitute | |
| # in 2* what we match, +1 more, plus the quote. | |
| windows_quoter_regex = re.compile(r'(\\*)"') | |
| arg = windows_quoter_regex.sub(lambda mo: 2 * mo.group(1) + '\\"', arg) | |
| # %'s also need to be doubled otherwise they're interpreted as batch | |
| # positional arguments. Also make sure to escape the % so that they're | |
| # passed literally through escaping so they can be singled to just the | |
| # original %. Otherwise, trying to pass the literal representation that | |
| # looks like an environment variable to the shell (e.g. %PATH%) would fail. | |
| arg = arg.replace('%', '%%') | |
| # These commands are used in rsp files, so no escaping for the shell (via ^) | |
| # is necessary. | |
| # Finally, wrap the whole thing in quotes so that the above quote rule | |
| # applies and whitespace isn't a word break. | |
| return '"' + arg + '"' | |
| def Join(self, command): | |
| assert not isinstance(command, basestring) | |
| return ' '.join(self.MaybeQuoteArgument(argument) for argument in command) | |
| def And(self, *commands): | |
| return ' && '.join(_MaybeJoin(self, command) for command in commands) | |
| def Or(self, *commands): | |
| return ' || '.join(_MaybeJoin(self, command) for command in commands) |