summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2024-02-09 10:36:12 (GMT)
committerGitHub <noreply@github.com>2024-02-09 10:36:12 (GMT)
commit846fd721d518dda88a7d427ec3d2c03c45d9fa90 (patch)
tree4c6b5f8e5dcc88646b945c1532e1be8835399eec
parentc968dc7ff3041137bb702436ff944692dede1ad1 (diff)
downloadcpython-846fd721d518dda88a7d427ec3d2c03c45d9fa90.zip
cpython-846fd721d518dda88a7d427ec3d2c03c45d9fa90.tar.gz
cpython-846fd721d518dda88a7d427ec3d2c03c45d9fa90.tar.bz2
gh-115059: Flush the underlying write buffer in io.BufferedRandom.read1() (GH-115163)
-rw-r--r--Lib/test/test_io.py52
-rw-r--r--Misc/NEWS.d/next/Library/2024-02-08-13-26-14.gh-issue-115059.DqP9dr.rst1
-rw-r--r--Modules/_io/bufferedio.c10
3 files changed, 63 insertions, 0 deletions
diff --git a/Lib/test/test_io.py b/Lib/test/test_io.py
index 73669ec..a24579d 100644
--- a/Lib/test/test_io.py
+++ b/Lib/test/test_io.py
@@ -2497,6 +2497,28 @@ class BufferedRandomTest(BufferedReaderTest, BufferedWriterTest):
f.flush()
self.assertEqual(raw.getvalue(), b'a2c')
+ def test_read1_after_write(self):
+ with self.BytesIO(b'abcdef') as raw:
+ with self.tp(raw, 3) as f:
+ f.write(b"1")
+ self.assertEqual(f.read1(1), b'b')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'1bcdef')
+ with self.BytesIO(b'abcdef') as raw:
+ with self.tp(raw, 3) as f:
+ f.write(b"1")
+ self.assertEqual(f.read1(), b'bcd')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'1bcdef')
+ with self.BytesIO(b'abcdef') as raw:
+ with self.tp(raw, 3) as f:
+ f.write(b"1")
+ # XXX: read(100) returns different numbers of bytes
+ # in Python and C implementations.
+ self.assertEqual(f.read1(100)[:3], b'bcd')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'1bcdef')
+
def test_interleaved_readline_write(self):
with self.BytesIO(b'ab\ncdef\ng\n') as raw:
with self.tp(raw) as f:
@@ -2509,6 +2531,36 @@ class BufferedRandomTest(BufferedReaderTest, BufferedWriterTest):
f.flush()
self.assertEqual(raw.getvalue(), b'1b\n2def\n3\n')
+ def test_xxx(self):
+ with self.BytesIO(b'abcdefgh') as raw:
+ with self.tp(raw) as f:
+ f.write(b'123')
+ self.assertEqual(f.read(), b'defgh')
+ f.write(b'456')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'123defgh456')
+ with self.BytesIO(b'abcdefgh') as raw:
+ with self.tp(raw) as f:
+ f.write(b'123')
+ self.assertEqual(f.read(3), b'def')
+ f.write(b'456')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'123def456')
+ with self.BytesIO(b'abcdefgh') as raw:
+ with self.tp(raw) as f:
+ f.write(b'123')
+ self.assertEqual(f.read1(), b'defgh')
+ f.write(b'456')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'123defgh456')
+ with self.BytesIO(b'abcdefgh') as raw:
+ with self.tp(raw) as f:
+ f.write(b'123')
+ self.assertEqual(f.read1(3), b'def')
+ f.write(b'456')
+ f.flush()
+ self.assertEqual(raw.getvalue(), b'123def456')
+
# You can't construct a BufferedRandom over a non-seekable stream.
test_unseekable = None
diff --git a/Misc/NEWS.d/next/Library/2024-02-08-13-26-14.gh-issue-115059.DqP9dr.rst b/Misc/NEWS.d/next/Library/2024-02-08-13-26-14.gh-issue-115059.DqP9dr.rst
new file mode 100644
index 0000000..331baed
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2024-02-08-13-26-14.gh-issue-115059.DqP9dr.rst
@@ -0,0 +1 @@
+:meth:`io.BufferedRandom.read1` now flushes the underlying write buffer.
diff --git a/Modules/_io/bufferedio.c b/Modules/_io/bufferedio.c
index f02207a..8ebe9ec 100644
--- a/Modules/_io/bufferedio.c
+++ b/Modules/_io/bufferedio.c
@@ -1050,6 +1050,16 @@ _io__Buffered_read1_impl(buffered *self, Py_ssize_t n)
Py_DECREF(res);
return NULL;
}
+ /* Flush the write buffer if necessary */
+ if (self->writable) {
+ PyObject *r = buffered_flush_and_rewind_unlocked(self);
+ if (r == NULL) {
+ LEAVE_BUFFERED(self)
+ Py_DECREF(res);
+ return NULL;
+ }
+ Py_DECREF(r);
+ }
_bufferedreader_reset_buf(self);
r = _bufferedreader_raw_read(self, PyBytes_AS_STRING(res), n);
LEAVE_BUFFERED(self)