summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMartin Panter <vadmium+py@gmail.com>2016-12-11 00:18:36 (GMT)
committerMartin Panter <vadmium+py@gmail.com>2016-12-11 00:18:36 (GMT)
commitca3263c50c3b3a3719d2ec3ee7b30b8c669dcb19 (patch)
treee18938b9e8d57f43e11c013c5d633fd64d9aca65
parentb94eef2ae3737b1f03452a30d5c96640c838dc46 (diff)
downloadcpython-ca3263c50c3b3a3719d2ec3ee7b30b8c669dcb19.zip
cpython-ca3263c50c3b3a3719d2ec3ee7b30b8c669dcb19.tar.gz
cpython-ca3263c50c3b3a3719d2ec3ee7b30b8c669dcb19.tar.bz2
Issue #25677: Correct syntax error caret for indented blocks.
Based on patch by Michael Layzell.
-rw-r--r--Lib/test/test_cmd_line_script.py33
-rw-r--r--Misc/ACKS1
-rw-r--r--Misc/NEWS3
-rw-r--r--Python/errors.c5
-rw-r--r--Python/pythonrun.c2
5 files changed, 39 insertions, 5 deletions
diff --git a/Lib/test/test_cmd_line_script.py b/Lib/test/test_cmd_line_script.py
index befe0e4..919f361 100644
--- a/Lib/test/test_cmd_line_script.py
+++ b/Lib/test/test_cmd_line_script.py
@@ -10,6 +10,7 @@ import os
import os.path
import py_compile
import subprocess
+import io
import textwrap
from test import support
@@ -540,6 +541,38 @@ class CmdLineTest(unittest.TestCase):
text = stderr.decode('ascii')
self.assertEqual(text, "some text")
+ def test_syntaxerror_unindented_caret_position(self):
+ script = "1 + 1 = 2\n"
+ with support.temp_dir() as script_dir:
+ script_name = _make_test_script(script_dir, 'script', script)
+ exitcode, stdout, stderr = assert_python_failure(script_name)
+ text = io.TextIOWrapper(io.BytesIO(stderr), 'ascii').read()
+ # Confirm that the caret is located under the first 1 character
+ self.assertIn("\n 1 + 1 = 2\n ^", text)
+
+ def test_syntaxerror_indented_caret_position(self):
+ script = textwrap.dedent("""\
+ if True:
+ 1 + 1 = 2
+ """)
+ with support.temp_dir() as script_dir:
+ script_name = _make_test_script(script_dir, 'script', script)
+ exitcode, stdout, stderr = assert_python_failure(script_name)
+ text = io.TextIOWrapper(io.BytesIO(stderr), 'ascii').read()
+ # Confirm that the caret is located under the first 1 character
+ self.assertIn("\n 1 + 1 = 2\n ^", text)
+
+ # Try the same with a form feed at the start of the indented line
+ script = (
+ "if True:\n"
+ "\f 1 + 1 = 2\n"
+ )
+ script_name = _make_test_script(script_dir, "script", script)
+ exitcode, stdout, stderr = assert_python_failure(script_name)
+ text = io.TextIOWrapper(io.BytesIO(stderr), "ascii").read()
+ self.assertNotIn("\f", text)
+ self.assertIn("\n 1 + 1 = 2\n ^", text)
+
def test_main():
support.run_unittest(CmdLineTest)
diff --git a/Misc/ACKS b/Misc/ACKS
index a9cd72e..e4b86390 100644
--- a/Misc/ACKS
+++ b/Misc/ACKS
@@ -840,6 +840,7 @@ Julia Lawall
Chris Lawrence
Mark Lawrence
Chris Laws
+Michael Layzell
Michael Lazar
Brian Leair
Mathieu Leduc-Hamel
diff --git a/Misc/NEWS b/Misc/NEWS
index 13e9e6d..f613d0d 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -10,6 +10,9 @@ Release date: TBA
Core and Builtins
-----------------
+- Issue #25677: Correct the positioning of the syntax error caret for
+ indented blocks. Based on patch by Michael Layzell.
+
- Issue #29000: Fixed bytes formatting of octals with zero padding in alternate
form.
diff --git a/Python/errors.c b/Python/errors.c
index ebfb3fd..b820722 100644
--- a/Python/errors.c
+++ b/Python/errors.c
@@ -1094,11 +1094,8 @@ err_programtext(FILE *fp, int lineno)
}
fclose(fp);
if (i == lineno) {
- char *p = linebuf;
PyObject *res;
- while (*p == ' ' || *p == '\t' || *p == '\014')
- p++;
- res = PyUnicode_FromString(p);
+ res = PyUnicode_FromString(linebuf);
if (res == NULL)
PyErr_Clear();
return res;
diff --git a/Python/pythonrun.c b/Python/pythonrun.c
index 72b6c9b..90cb2de 100644
--- a/Python/pythonrun.c
+++ b/Python/pythonrun.c
@@ -528,7 +528,7 @@ print_error_text(PyObject *f, int offset, PyObject *text_obj)
offset -= (int)(nl+1-text);
text = nl+1;
}
- while (*text == ' ' || *text == '\t') {
+ while (*text == ' ' || *text == '\t' || *text == '\f') {
text++;
offset--;
}