Commit ba42fd58 authored by Ezio Melotti's avatar Ezio Melotti

#6780: fix starts/endswith error message to mention that tuples are accepted too.

parent 2043f9c5
...@@ -290,6 +290,14 @@ class BaseBytesTest(unittest.TestCase): ...@@ -290,6 +290,14 @@ class BaseBytesTest(unittest.TestCase):
self.assertTrue(b.startswith(b"h")) self.assertTrue(b.startswith(b"h"))
self.assertFalse(b.startswith(b"hellow")) self.assertFalse(b.startswith(b"hellow"))
self.assertFalse(b.startswith(b"ha")) self.assertFalse(b.startswith(b"ha"))
try:
b.startswith([b'h'])
except TypeError as err:
exc = str(err)
else:
self.fail('startswith unexpectedly succeeded')
self.assertIn('bytes', exc)
self.assertIn('tuple', exc)
def test_endswith(self): def test_endswith(self):
b = self.type2test(b'hello') b = self.type2test(b'hello')
...@@ -299,6 +307,14 @@ class BaseBytesTest(unittest.TestCase): ...@@ -299,6 +307,14 @@ class BaseBytesTest(unittest.TestCase):
self.assertTrue(b.endswith(b"o")) self.assertTrue(b.endswith(b"o"))
self.assertFalse(b.endswith(b"whello")) self.assertFalse(b.endswith(b"whello"))
self.assertFalse(b.endswith(b"no")) self.assertFalse(b.endswith(b"no"))
try:
b.endswith([b'o'])
except TypeError as err:
exc = str(err)
else:
self.fail('endswith unexpectedly succeeded')
self.assertIn('bytes', exc)
self.assertIn('tuple', exc)
def test_find(self): def test_find(self):
b = self.type2test(b'mississippi') b = self.type2test(b'mississippi')
......
...@@ -789,6 +789,17 @@ class UnicodeTest( ...@@ -789,6 +789,17 @@ class UnicodeTest(
self.assertEqual('%f' % INF, 'inf') self.assertEqual('%f' % INF, 'inf')
self.assertEqual('%F' % INF, 'INF') self.assertEqual('%F' % INF, 'INF')
def test_startswith_endswith_errors(self):
for meth in ('foo'.startswith, 'foo'.endswith):
try:
meth(['f'])
except TypeError as err:
exc = str(err)
else:
self.fail('starts/endswith unexpectedly succeeded')
self.assertIn('str', exc)
self.assertIn('tuple', exc)
@support.run_with_locale('LC_ALL', 'de_DE', 'fr_FR') @support.run_with_locale('LC_ALL', 'de_DE', 'fr_FR')
def test_format_float(self): def test_format_float(self):
# should not format with a comma, but always with C locale # should not format with a comma, but always with C locale
......
...@@ -10,6 +10,9 @@ What's New in Python 3.1.4? ...@@ -10,6 +10,9 @@ What's New in Python 3.1.4?
Core and Builtins Core and Builtins
----------------- -----------------
- Issue #6780: fix starts/endswith error message to mention that tuples are
accepted too.
- Issue #5057: fix a bug in the peepholer that led to non-portable pyc files - Issue #5057: fix a bug in the peepholer that led to non-portable pyc files
between narrow and wide builds while optimizing BINARY_SUBSCR on non-BMP between narrow and wide builds while optimizing BINARY_SUBSCR on non-BMP
chars (e.g. "\U00012345"[0]). chars (e.g. "\U00012345"[0]).
......
...@@ -1281,7 +1281,7 @@ PyDoc_STRVAR(startswith__doc__, ...@@ -1281,7 +1281,7 @@ PyDoc_STRVAR(startswith__doc__,
Return True if B starts with the specified prefix, False otherwise.\n\ Return True if B starts with the specified prefix, False otherwise.\n\
With optional start, test B beginning at that position.\n\ With optional start, test B beginning at that position.\n\
With optional end, stop comparing B at that position.\n\ With optional end, stop comparing B at that position.\n\
prefix can also be a tuple of strings to try."); prefix can also be a tuple of bytes to try.");
static PyObject * static PyObject *
bytearray_startswith(PyByteArrayObject *self, PyObject *args) bytearray_startswith(PyByteArrayObject *self, PyObject *args)
...@@ -1308,8 +1308,12 @@ bytearray_startswith(PyByteArrayObject *self, PyObject *args) ...@@ -1308,8 +1308,12 @@ bytearray_startswith(PyByteArrayObject *self, PyObject *args)
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
result = _bytearray_tailmatch(self, subobj, start, end, -1); result = _bytearray_tailmatch(self, subobj, start, end, -1);
if (result == -1) if (result == -1) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "startswith first arg must be bytes "
"or a tuple of bytes, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
else else
return PyBool_FromLong(result); return PyBool_FromLong(result);
} }
...@@ -1320,7 +1324,7 @@ PyDoc_STRVAR(endswith__doc__, ...@@ -1320,7 +1324,7 @@ PyDoc_STRVAR(endswith__doc__,
Return True if B ends with the specified suffix, False otherwise.\n\ Return True if B ends with the specified suffix, False otherwise.\n\
With optional start, test B beginning at that position.\n\ With optional start, test B beginning at that position.\n\
With optional end, stop comparing B at that position.\n\ With optional end, stop comparing B at that position.\n\
suffix can also be a tuple of strings to try."); suffix can also be a tuple of bytes to try.");
static PyObject * static PyObject *
bytearray_endswith(PyByteArrayObject *self, PyObject *args) bytearray_endswith(PyByteArrayObject *self, PyObject *args)
...@@ -1347,8 +1351,12 @@ bytearray_endswith(PyByteArrayObject *self, PyObject *args) ...@@ -1347,8 +1351,12 @@ bytearray_endswith(PyByteArrayObject *self, PyObject *args)
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
result = _bytearray_tailmatch(self, subobj, start, end, +1); result = _bytearray_tailmatch(self, subobj, start, end, +1);
if (result == -1) if (result == -1) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "endswith first arg must be bytes or "
"a tuple of bytes, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
else else
return PyBool_FromLong(result); return PyBool_FromLong(result);
} }
......
...@@ -2654,8 +2654,12 @@ bytes_startswith(PyBytesObject *self, PyObject *args) ...@@ -2654,8 +2654,12 @@ bytes_startswith(PyBytesObject *self, PyObject *args)
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
result = _bytes_tailmatch(self, subobj, start, end, -1); result = _bytes_tailmatch(self, subobj, start, end, -1);
if (result == -1) if (result == -1) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "startswith first arg must be bytes "
"or a tuple of bytes, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
else else
return PyBool_FromLong(result); return PyBool_FromLong(result);
} }
...@@ -2694,8 +2698,12 @@ bytes_endswith(PyBytesObject *self, PyObject *args) ...@@ -2694,8 +2698,12 @@ bytes_endswith(PyBytesObject *self, PyObject *args)
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
result = _bytes_tailmatch(self, subobj, start, end, +1); result = _bytes_tailmatch(self, subobj, start, end, +1);
if (result == -1) if (result == -1) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "endswith first arg must be bytes or "
"a tuple of bytes, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
else else
return PyBool_FromLong(result); return PyBool_FromLong(result);
} }
......
...@@ -8743,8 +8743,12 @@ unicode_startswith(PyUnicodeObject *self, ...@@ -8743,8 +8743,12 @@ unicode_startswith(PyUnicodeObject *self,
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
substring = (PyUnicodeObject *)PyUnicode_FromObject(subobj); substring = (PyUnicodeObject *)PyUnicode_FromObject(subobj);
if (substring == NULL) if (substring == NULL) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "startswith first arg must be str or "
"a tuple of str, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
result = tailmatch(self, substring, start, end, -1); result = tailmatch(self, substring, start, end, -1);
Py_DECREF(substring); Py_DECREF(substring);
return PyBool_FromLong(result); return PyBool_FromLong(result);
...@@ -8787,9 +8791,12 @@ unicode_endswith(PyUnicodeObject *self, ...@@ -8787,9 +8791,12 @@ unicode_endswith(PyUnicodeObject *self,
Py_RETURN_FALSE; Py_RETURN_FALSE;
} }
substring = (PyUnicodeObject *)PyUnicode_FromObject(subobj); substring = (PyUnicodeObject *)PyUnicode_FromObject(subobj);
if (substring == NULL) if (substring == NULL) {
if (PyErr_ExceptionMatches(PyExc_TypeError))
PyErr_Format(PyExc_TypeError, "endswith first arg must be str or "
"a tuple of str, not %s", Py_TYPE(subobj)->tp_name);
return NULL; return NULL;
}
result = tailmatch(self, substring, start, end, +1); result = tailmatch(self, substring, start, end, +1);
Py_DECREF(substring); Py_DECREF(substring);
return PyBool_FromLong(result); return PyBool_FromLong(result);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment