summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAndrew Nester <andrew.nester.dev@gmail.com>2017-04-04 10:46:25 (GMT)
committerSerhiy Storchaka <storchaka@gmail.com>2017-04-04 10:46:25 (GMT)
commitf78b119364b521307237a1484ba5f43f42300898 (patch)
tree240db0104a9eba34ef4d5bd89ad9cb13472f97bc
parent5de85a17029356084b96db63e04d9eb150efd9c0 (diff)
downloadcpython-f78b119364b521307237a1484ba5f43f42300898.zip
cpython-f78b119364b521307237a1484ba5f43f42300898.tar.gz
cpython-f78b119364b521307237a1484ba5f43f42300898.tar.bz2
bpo-29649: Improve struct.pack_into() boundary error messages (#424)
-rw-r--r--Lib/test/test_struct.py20
-rw-r--r--Misc/NEWS3
-rw-r--r--Modules/_struct.c34
3 files changed, 53 insertions, 4 deletions
diff --git a/Lib/test/test_struct.py b/Lib/test/test_struct.py
index cf1d567..932ef47 100644
--- a/Lib/test/test_struct.py
+++ b/Lib/test/test_struct.py
@@ -578,6 +578,26 @@ class StructTest(unittest.TestCase):
self.check_sizeof('0s', 1)
self.check_sizeof('0c', 0)
+ def test_boundary_error_message(self):
+ regex = (
+ r'pack_into requires a buffer of at least 6 '
+ r'bytes for packing 1 bytes at offset 5 '
+ r'\(actual buffer size is 1\)'
+ )
+ with self.assertRaisesRegex(struct.error, regex):
+ struct.pack_into('b', bytearray(1), 5, 1)
+
+ def test_boundary_error_message_with_negative_offset(self):
+ byte_list = bytearray(10)
+ with self.assertRaisesRegex(
+ struct.error,
+ r'no space to pack 4 bytes at offset -2'):
+ struct.pack_into('<I', byte_list, -2, 123)
+
+ with self.assertRaisesRegex(
+ struct.error,
+ 'offset -11 out of range for 10-byte buffer'):
+ struct.pack_into('<B', byte_list, -11, 123)
class UnpackIteratorTest(unittest.TestCase):
"""
diff --git a/Misc/NEWS b/Misc/NEWS
index 2e9cce6..95092af 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -303,6 +303,9 @@ Extension Modules
Library
-------
+- bpo-29649: Improve struct.pack_into() exception messages for problems
+ with the buffer size and offset. Patch by Andrew Nester.
+
- bpo-29654: Support If-Modified-Since HTTP header (browser cache). Patch
by Pierre Quentel.
diff --git a/Modules/_struct.c b/Modules/_struct.c
index f66ee18..4bc4186 100644
--- a/Modules/_struct.c
+++ b/Modules/_struct.c
@@ -1931,14 +1931,40 @@ s_pack_into(PyObject *self, PyObject **args, Py_ssize_t nargs, PyObject *kwnames
}
/* Support negative offsets. */
- if (offset < 0)
+ if (offset < 0) {
+ /* Check that negative offset is low enough to fit data */
+ if (offset + soself->s_size > 0) {
+ PyErr_Format(StructError,
+ "no space to pack %zd bytes at offset %zd",
+ soself->s_size,
+ offset);
+ PyBuffer_Release(&buffer);
+ return NULL;
+ }
+
+ /* Check that negative offset is not crossing buffer boundary */
+ if (offset + buffer.len < 0) {
+ PyErr_Format(StructError,
+ "offset %zd out of range for %zd-byte buffer",
+ offset,
+ buffer.len);
+ PyBuffer_Release(&buffer);
+ return NULL;
+ }
+
offset += buffer.len;
+ }
/* Check boundaries */
- if (offset < 0 || (buffer.len - offset) < soself->s_size) {
+ if ((buffer.len - offset) < soself->s_size) {
PyErr_Format(StructError,
- "pack_into requires a buffer of at least %zd bytes",
- soself->s_size);
+ "pack_into requires a buffer of at least %zd bytes for "
+ "packing %zd bytes at offset %zd "
+ "(actual buffer size is %zd)",
+ soself->s_size + offset,
+ soself->s_size,
+ offset,
+ buffer.len);
PyBuffer_Release(&buffer);
return NULL;
}