py/modio: Fix the case where write fails in BufferedWriter.flush.

Previously, there was no test coverage of the "write failed" path.  In
fact, the assertion would fire instead of gracefully raising a Python
exception.

Slightly re-organize the code to place the assertion later.  Add a test
case which exercises all paths, and update the expected output.

Signed-off-by: Jeff Epler <jepler@gmail.com>
This commit is contained in:
Jeff Epler
2025-06-13 17:24:08 +02:00
committed by Damien George
parent 5ade8b7058
commit b6b7d64bd9
3 changed files with 34 additions and 4 deletions

View File

@@ -169,12 +169,13 @@ static mp_obj_t bufwriter_flush(mp_obj_t self_in) {
int err;
mp_uint_t out_sz = mp_stream_write_exactly(self->stream, self->buf, self->len, &err);
(void)out_sz;
// TODO: try to recover from a case of non-blocking stream, e.g. move
// remaining chunk to the beginning of buffer.
assert(out_sz == self->len);
self->len = 0;
if (err != 0) {
mp_raise_OSError(err);
} else {
// TODO: try to recover from a case of non-blocking stream, e.g. move
// remaining chunk to the beginning of buffer.
assert(out_sz == self->len);
self->len = 0;
}
}

View File

@@ -28,3 +28,27 @@ print(bts.getvalue())
# hashing a BufferedWriter
print(type(hash(buf)))
# Test failing flush()
class MyIO(io.IOBase):
def __init__(self):
self.count = 0
def write(self, buf):
self.count += 1
if self.count < 3:
return None
print("writing", buf)
return len(buf)
buf = io.BufferedWriter(MyIO(), 8)
buf.write(b"foobar")
for _ in range(4):
try:
buf.flush()
print("flushed")
except OSError:
print("OSError")

View File

@@ -4,3 +4,8 @@ b'foobarfoobar'
b'foobarfoobar'
b'foo'
<class 'int'>
OSError
OSError
writing bytearray(b'foobar')
flushed
flushed