Index: docs/CommandGuide/lit.rst =================================================================== --- docs/CommandGuide/lit.rst +++ docs/CommandGuide/lit.rst @@ -85,6 +85,10 @@ Echo all commands to stdout, as they are being executed. This can be valuable for debugging test failures, as the last echoed command will be the one which has failed. + :program:`lit` normally inserts a null command (``:`` in the case of bash) + with argument ``'RUN: at line N'`` before each command pipeline, and this + option also causes those null commands to be echoed to stdout to help you + locate the source line of the failed command. This option implies ``--verbose``. .. option:: -a, --show-all Index: utils/lit/lit/TestRunner.py =================================================================== --- utils/lit/lit/TestRunner.py +++ utils/lit/lit/TestRunner.py @@ -40,6 +40,17 @@ kAvoidDevNull = kIsWindows kDevNull = "/dev/null" +# A regex that matches %dbg(ARG), which lit inserts at the beginning of each +# run command pipeline such that ARG specifies the pipeline's source line +# number. lit later expands each %dbg(ARG) to a command that behaves as a null +# command in the target shell so that the line number is seen in lit's verbose +# mode. +# +# This regex captures ARG. ARG must not contain a right parenthesis, which +# terminates %dbg. ARG must not contain quotes, in which ARG might be enclosed +# during expansion. +kPdbgRegex = '%dbg\(([^)\'"]*)\)' + class ShellEnvironment(object): """Mutable shell environment containing things like CWD and env vars. @@ -789,6 +800,13 @@ results.append(cmdResult) return cmdResult.exitCode + if cmd.commands[0].args[0] == ':': + if len(cmd.commands) != 1: + raise InternalShellError(cmd.commands[0], "Unsupported: ':' " + "cannot be part of a pipeline") + results.append(ShellCommandResult(cmd.commands[0], '', '', 0, False)) + return 0; + procs = [] default_stdin = subprocess.PIPE stderrTempFiles = [] @@ -981,6 +999,7 @@ return exitCode def executeScriptInternal(test, litConfig, tmpBase, commands, cwd): + commands = applySubstitutions(commands, [(kPdbgRegex, ": '\\1'")]) cmds = [] for ln in commands: try: @@ -1051,7 +1070,7 @@ out += 'error: command reached timeout: %s\n' % ( str(result.timeoutReached),) - return out, err, exitCode, timeoutInfo + return out, err, exitCode, timeoutInfo, commands def executeScript(test, litConfig, tmpBase, commands, cwd): bashPath = litConfig.getBashPath() @@ -1066,9 +1085,12 @@ mode += 'b' # Avoid CRLFs when writing bash scripts. f = open(script, mode) if isWin32CMDEXE: + commands = applySubstitutions(commands, [(kPdbgRegex, + "echo '\\1' > nul")]) f.write('@echo off\n') f.write('\nif %ERRORLEVEL% NEQ 0 EXIT\n'.join(commands)) else: + commands = applySubstitutions(commands, [(kPdbgRegex, ": '\\1'")]) if test.config.pipefail: f.write('set -o pipefail;') if litConfig.echo_all_commands: @@ -1093,9 +1115,9 @@ out, err, exitCode = lit.util.executeCommand(command, cwd=cwd, env=test.config.environment, timeout=litConfig.maxIndividualTestTime) - return (out, err, exitCode, None) + return (out, err, exitCode, None, commands) except lit.util.ExecuteCommandTimeoutException as e: - return (e.out, e.err, e.exitCode, e.msg) + return (e.out, e.err, e.exitCode, e.msg, commands) def parseIntegratedTestScriptCommands(source_path, keywords): """ @@ -1215,14 +1237,14 @@ ]) return substitutions -def applySubstitutions(script, substitutions): +def applySubstitutions(script, substitutions, escapeForWindows=False): """Apply substitutions to the script. Allow full regular expression syntax. Replace each matching occurrence of regular expression pattern a with substitution b in line ln.""" def processLine(ln): # Apply substitutions for a,b in substitutions: - if kIsWindows: + if kIsWindows and escapeForWindows: b = b.replace("\\","\\\\") ln = re.sub(a, b, ln) @@ -1301,7 +1323,9 @@ self.parser = parser if kind == ParserKind.COMMAND: - self.parser = self._handleCommand + self.parser = lambda line_number, line, output: \ + self._handleCommand(line_number, line, output, + self.keyword) elif kind == ParserKind.LIST: self.parser = self._handleList elif kind == ParserKind.BOOLEAN_EXPR: @@ -1332,7 +1356,7 @@ return (not line.strip() or output) @staticmethod - def _handleCommand(line_number, line, output): + def _handleCommand(line_number, line, output, keyword): """A helper for parsing COMMAND type keywords""" # Trim trailing whitespace. line = line.rstrip() @@ -1351,6 +1375,14 @@ else: if output is None: output = [] + pdbg = "%dbg({keyword} at line {line_number})".format( + keyword=keyword, + line_number=line_number) + assert re.match(kPdbgRegex + "$", pdbg), \ + "kPdbgRegex expected to match actual %dbg usage" + line = "{pdbg} && {real_command}".format( + pdbg=pdbg, + real_command=line) output.append(line) return output @@ -1489,7 +1521,7 @@ if isinstance(res, lit.Test.Result): return res - out,err,exitCode,timeoutInfo = res + out,err,exitCode,timeoutInfo,script = res if exitCode == 0: status = Test.PASS else: @@ -1530,7 +1562,7 @@ substitutions = list(extra_substitutions) substitutions += getDefaultSubstitutions(test, tmpDir, tmpBase, normalize_slashes=useExternalSh) - script = applySubstitutions(script, substitutions) + script = applySubstitutions(script, substitutions, True) # Re-run failed tests up to test_retry_attempts times. attempts = 1 Index: utils/lit/tests/Inputs/shtest-run-at-line/external-shell/basic.txt =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/external-shell/basic.txt @@ -0,0 +1,8 @@ +# These commands must run under both bash and windows cmd.exe. + +# RUN: echo bad > %t +# RUN: FileCheck --input-file %t %s +# RUN: echo good > %t +# RUN: FileCheck --input-file %t %s + +# CHECK: good Index: utils/lit/tests/Inputs/shtest-run-at-line/external-shell/line-continuation.txt =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/external-shell/line-continuation.txt @@ -0,0 +1,11 @@ +# These commands must run under both bash and windows cmd.exe. + +# RUN: echo 'foo bar' \ +# RUN: | FileCheck %s +# RUN: echo \ +# RUN: 'foo baz' \ +# RUN: | FileCheck %s +# RUN: echo 'foo bar' \ +# RUN: | FileCheck %s + +# CHECK: foo bar Index: utils/lit/tests/Inputs/shtest-run-at-line/external-shell/lit.local.cfg =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/external-shell/lit.local.cfg @@ -0,0 +1,2 @@ +import lit.formats +config.test_format = lit.formats.ShTest(execute_external=True) Index: utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/basic.txt =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/basic.txt @@ -0,0 +1,3 @@ +# RUN: true +# RUN: false +# RUN: true Index: utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/line-continuation.txt =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/line-continuation.txt @@ -0,0 +1,11 @@ +# RUN: : first line continued \ +# RUN: to second line +# RUN: echo 'foo bar' \ +# RUN: | FileCheck %s +# RUN: echo \ +# RUN: 'foo baz' \ +# RUN: | FileCheck %s +# RUN: echo 'foo bar' \ +# RUN: | FileCheck %s + +# CHECK: foo bar Index: utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/lit.local.cfg =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/internal-shell/lit.local.cfg @@ -0,0 +1,2 @@ +import lit.formats +config.test_format = lit.formats.ShTest(execute_external=False) Index: utils/lit/tests/Inputs/shtest-run-at-line/lit.cfg =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-run-at-line/lit.cfg @@ -0,0 +1,2 @@ +config.name = 'shtest-run-at-line' +config.suffixes = ['.txt'] Index: utils/lit/tests/Inputs/shtest-shell/colon-error.txt =================================================================== --- /dev/null +++ utils/lit/tests/Inputs/shtest-shell/colon-error.txt @@ -0,0 +1,3 @@ +# Check error on an unsupported ":". (cannot be part of a pipeline) +# +# RUN: : | echo "hello" Index: utils/lit/tests/lit.cfg =================================================================== --- utils/lit/tests/lit.cfg +++ utils/lit/tests/lit.cfg @@ -1,6 +1,7 @@ # -*- Python -*- import os +import platform import sys import lit.formats @@ -71,3 +72,12 @@ if directory: path = os.path.pathsep.join((directory, path)) config.environment['PATH'] = path + +# substitutions will be expanded in double quotes +isWin32CMDEXE = platform.system() == 'Windows' and not config.getBashPath() +if isWin32CMDEXE: + config.substitutions.append(('%{pdbg0}', "echo '")) + config.substitutions.append(('%{pdbg1}', "' > /nul")) +else: + config.substitutions.append(('%{pdbg0}', ": '")) + config.substitutions.append(('%{pdbg1}', "'")) Index: utils/lit/tests/max-failures.py =================================================================== --- utils/lit/tests/max-failures.py +++ utils/lit/tests/max-failures.py @@ -8,7 +8,7 @@ # # END. -# CHECK: Failing Tests (26) +# CHECK: Failing Tests (27) # CHECK: Failing Tests (1) # CHECK: Failing Tests (2) # CHECK: error: Setting --max-failures to 0 does not have any effect. Index: utils/lit/tests/shtest-format.py =================================================================== --- utils/lit/tests/shtest-format.py +++ utils/lit/tests/shtest-format.py @@ -39,6 +39,7 @@ # # CHECK: Command Output (stdout): # CHECK-NEXT: -- +# CHECK-NEXT: $ ":" "RUN: at line 1" # CHECK-NEXT: $ "printf" # CHECK-NEXT: # command output: # CHECK-NEXT: line 1: failed test output on stdout Index: utils/lit/tests/shtest-output-printing.py =================================================================== --- utils/lit/tests/shtest-output-printing.py +++ utils/lit/tests/shtest-output-printing.py @@ -16,12 +16,15 @@ # # CHECK: Command Output # CHECK-NEXT: -- +# CHECK-NEXT: $ ":" "RUN: at line 1" # CHECK-NEXT: $ "true" +# CHECK-NEXT: $ ":" "RUN: at line 2" # CHECK-NEXT: $ "echo" "hi" # CHECK-NEXT: # command output: # CHECK-NEXT: hi # -# CHECK: $ "wc" "missing-file" +# CHECK: $ ":" "RUN: at line 3" +# CHECK-NEXT: $ "wc" "missing-file" # CHECK-NEXT: # redirected output from '{{.*(/|\\\\)}}basic.txt.tmp.out': # CHECK-NEXT: missing-file{{.*}} No such file or directory # CHECK: note: command had no output on stdout or stderr Index: utils/lit/tests/shtest-run-at-line.py =================================================================== --- /dev/null +++ utils/lit/tests/shtest-run-at-line.py @@ -0,0 +1,73 @@ +# Check that -vv makes the line number of the failing RUN command clear. +# (-v is actually sufficient in the case of the internal shell.) +# +# RUN: not %{lit} -j 1 -vv %{inputs}/shtest-run-at-line > %t.out +# RUN: FileCheck --input-file %t.out -Dpdbg0="%{pdbg0}" -Dpdbg1="%{pdbg1}" %s +# +# END. + + +# CHECK: Testing: 4 tests + + +# In the case of the external shell, we check for only RUN lines in stderr in +# case some shell implementations format "set -x" output differently. + +# CHECK-LABEL: FAIL: shtest-run-at-line :: external-shell/basic.txt + +# CHECK: Script: +# CHECK: [[pdbg0]]RUN: at line 3[[pdbg1]] && echo bad > +# CHECK-NEXT: [[pdbg0]]RUN: at line 4[[pdbg1]] && FileCheck --input-file +# CHECK: [[pdbg0]]RUN: at line 5[[pdbg1]] && echo good > +# CHECK-NEXT: [[pdbg0]]RUN: at line 6[[pdbg1]] && FileCheck --input-file + +# CHECK: Command Output (stderr) +# CHECK: RUN: at line 3 +# CHECK: RUN: at line 4 +# CHECK-NOT: RUN + +# CHECK-LABEL: FAIL: shtest-run-at-line :: external-shell/line-continuation.txt + +# CHECK: Script: +# CHECK: [[pdbg0]]RUN: at line 3[[pdbg1]] && echo 'foo bar' | FileCheck +# CHECK-NEXT: [[pdbg0]]RUN: at line 5[[pdbg1]] && echo 'foo baz' | FileCheck +# CHECK-NEXT: [[pdbg0]]RUN: at line 8[[pdbg1]] && echo 'foo bar' | FileCheck + +# CHECK: Command Output (stderr) +# CHECK: RUN: at line 3 +# CHECK: RUN: at line 5 +# CHECK-NOT: RUN + + +# CHECK-LABEL: FAIL: shtest-run-at-line :: internal-shell/basic.txt + +# CHECK: Script: +# CHECK: : 'RUN: at line 1' && true +# CHECK-NEXT: : 'RUN: at line 2' && false +# CHECK-NEXT: : 'RUN: at line 3' && true + +# CHECK: Command Output (stdout) +# CHECK: $ ":" "RUN: at line 1" +# CHECK-NEXT: $ "true" +# CHECK-NEXT: $ ":" "RUN: at line 2" +# CHECK-NEXT: $ "false" +# CHECK-NOT: RUN + +# CHECK-LABEL: FAIL: shtest-run-at-line :: internal-shell/line-continuation.txt + +# CHECK: Script: +# CHECK: : 'RUN: at line 1' && : first line continued to second line +# CHECK-NEXT: : 'RUN: at line 3' && echo 'foo bar' | FileCheck +# CHECK-NEXT: : 'RUN: at line 5' && echo 'foo baz' | FileCheck +# CHECK-NEXT: : 'RUN: at line 8' && echo 'foo bar' | FileCheck + +# CHECK: Command Output (stdout) +# CHECK: $ ":" "RUN: at line 1" +# CHECK-NEXT: $ ":" "first" "line" "continued" "to" "second" "line" +# CHECK-NEXT: $ ":" "RUN: at line 3" +# CHECK-NEXT: $ "echo" "foo bar" +# CHECK-NEXT: $ "FileCheck" "{{.*}}" +# CHECK-NEXT: $ ":" "RUN: at line 5" +# CHECK-NEXT: $ "echo" "foo baz" +# CHECK-NEXT: $ "FileCheck" "{{.*}}" +# CHECK-NOT: RUN Index: utils/lit/tests/shtest-shell.py =================================================================== --- utils/lit/tests/shtest-shell.py +++ utils/lit/tests/shtest-shell.py @@ -26,6 +26,14 @@ # CHECK: error: command failed with exit status: 1 # CHECK: *** +# CHECK: FAIL: shtest-shell :: colon-error.txt +# CHECK: *** TEST 'shtest-shell :: colon-error.txt' FAILED *** +# CHECK: $ ":" +# CHECK: # command stderr: +# CHECK: Unsupported: ':' cannot be part of a pipeline +# CHECK: error: command failed with exit status: 127 +# CHECK: *** + # CHECK: FAIL: shtest-shell :: diff-error-0.txt # CHECK: *** TEST 'shtest-shell :: diff-error-0.txt' FAILED *** # CHECK: $ "diff" "diff-error-0.txt" "diff-error-0.txt" @@ -153,7 +161,7 @@ # # CHECK: FAIL: shtest-shell :: error-1.txt # CHECK: *** TEST 'shtest-shell :: error-1.txt' FAILED *** -# CHECK: shell parser error on: 'echo "missing quote' +# CHECK: shell parser error on: ': \'RUN: at line 3\' && echo "missing quote' # CHECK: *** # CHECK: FAIL: shtest-shell :: error-2.txt @@ -219,4 +227,4 @@ # CHECK: PASS: shtest-shell :: sequencing-0.txt # CHECK: XFAIL: shtest-shell :: sequencing-1.txt # CHECK: PASS: shtest-shell :: valid-shell.txt -# CHECK: Failing Tests (26) +# CHECK: Failing Tests (27) Index: utils/lit/tests/unit/TestRunner.py =================================================================== --- utils/lit/tests/unit/TestRunner.py +++ utils/lit/tests/unit/TestRunner.py @@ -99,8 +99,8 @@ cmd_parser = self.get_parser(parsers, 'MY_RUN:') value = cmd_parser.getValue() self.assertEqual(len(value), 2) # there are only two run lines - self.assertEqual(value[0].strip(), 'baz') - self.assertEqual(value[1].strip(), 'foo bar') + self.assertEqual(value[0].strip(), "%dbg(MY_RUN: at line 4) && baz") + self.assertEqual(value[1].strip(), "%dbg(MY_RUN: at line 7) && foo bar") def test_custom(self): parsers = self.make_parsers()