io — input/output streams¶
This module contains additional types of stream (file-like) objects
and helper functions. It exposes the open() builtin along with
in-memory text and binary buffers (StringIO, BytesIO)
that implement the standard read/write/seek stream interface.
Conceptual hierarchy¶
Difference to CPython
Conceptual hierarchy of stream base classes is simplified in MicroPython, as described in this section.
(Abstract) base stream classes, which serve as a foundation for behaviour of all the concrete classes, adhere to few dichotomies (pair-wise classifications) in CPython. In MicroPython, they are somewhat simplified and made implicit to achieve higher efficiencies and save resources.
An important dichotomy in CPython is unbuffered vs buffered streams. In MicroPython, all streams are currently unbuffered. This is because all modern OSes, and even many RTOSes and filesystem drivers already perform buffering on their side. Adding another layer of buffering is counter- productive (an issue known as “bufferbloat”) and takes precious memory. Note that there still cases where buffering may be useful, so we may introduce optional buffering support at a later time.
But in CPython, another important dichotomy is tied with “bufferedness” - it’s whether a stream may incur short read/writes or not. A short read is when a user asks e.g. 10 bytes from a stream, but gets less, similarly for writes. In CPython, unbuffered streams are automatically short operation susceptible, while buffered are guarantee against them. The no short read/writes is an important trait, as it allows to develop more concise and efficient programs - something which is highly desirable for MicroPython. So, while MicroPython doesn’t support buffered streams, it still provides for no-short-operations streams. Whether there will be short operations or not depends on each particular class’ needs, but developers are strongly advised to favour no-short-operations behaviour for the reasons stated above. For example, MicroPython sockets are guaranteed to avoid short read/writes. Actually, at this time, there is no example of a short-operations stream class in the core, and such a class would be specific to particular hardware.
The no-short-operations behaviour gets tricky in case of non-blocking streams, blocking vs non-blocking behaviour being another CPython dichotomy, fully supported by MicroPython. Non-blocking streams never wait for data either to arrive or be written - they read/write whatever possible, or signal lack of data (or ability to write data). Clearly, this conflicts with “no-short-operations” policy, and indeed, a case of non-blocking buffered (and this no-short-ops) streams is convoluted in CPython - in some places, such combination is prohibited, in some it’s undefined or just not documented, in some cases it raises verbose exceptions. The matter is much simpler in MicroPython: non-blocking stream are important for efficient asynchronous operations, so this property prevails on the “no-short-ops” one. So, while blocking streams will avoid short reads/writes whenever possible (the only case to get a short read is if end of file is reached, or in case of error (but errors don’t return short data, but raise exceptions)), non-blocking streams may produce short data to avoid blocking the operation.
The final dichotomy is binary vs text streams. MicroPython of course supports these, but while in CPython text streams are inherently buffered, they aren’t in MicroPython. (Indeed, that’s one of the cases for which we may introduce buffering support.)
Note that for efficiency, MicroPython doesn’t provide abstract base classes corresponding to the hierarchy above, and it’s not possible to implement, or subclass, a stream class in pure Python.
Functions¶
Classes¶
- class io.IOBase¶
Base class for stream (“file-like”) objects. Concrete subclasses implement the low-level I/O methods below (
readinto,write,ioctl); the runtime builds the higher-level stream protocol (read,readline,readlines,close, iteration) on top of them, so every stream instance supports those methods even when the subclass does not define them.Implementation methods (override these in a subclass):
- readinto(buf: bytearray) int | None¶
Read bytes into the writable buffer buf. Return the number of bytes read,
0at end of stream, orNoneif no data is available right now (for a non-blocking stream).
- write(buf: bytes) int | None¶
Write the bytes in buf. Return the number of bytes written, or
Noneif the write cannot be performed right now (for a non-blocking stream).
- ioctl(request: int, arg: int) int¶
Control the underlying stream/device. request is one of the
MP_STREAM_*request codes. Return a non-negative value on success, or a negativeerrnovalue on error.
Stream protocol methods (available on every stream instance):
- read(size: int = -1)¶
Read and return up to size bytes (or characters, in text mode). If size is omitted or negative, read until end of stream. Returns
bytesfor binary streams andstrfor text streams; an empty result indicates end of stream.
- readline(size: int = -1)¶
Read and return one line, including the trailing newline character if one is present. If size is given, at most size bytes (or characters) are read. Returns an empty
bytes/strat end of stream.
- readlines() list¶
Read until end of stream and return a
listof lines, each with its trailing newline.
- close() None¶
Close the stream and release any underlying resources. Operations on a closed stream raise
OSError(orValueErrorfor in-memory streams).
- seek(offset: int, whence: int = 0) int¶
Change the current stream position to offset bytes relative to whence (
0= start of stream,1= current position,2= end of stream). Return the new absolute position. RaisesOSErroron a stream that is not seekable.
- flush() None¶
Flush any write buffers, pushing pending data to the underlying device or file. A no-op on streams that do not buffer.
Iterating a stream directly yields one line per iteration – equivalent to calling
readline()in a loop until the empty-line end-of-stream sentinel is returned. A stream also supports the context-manager protocol, sowith open(...) as f:closes the stream automatically.Note
MicroPython’s stream module also exposes “1”-suffixed C helpers
mp_stream_read1_obj,mp_stream_readinto1_obj, andmp_stream_write1_objthat perform a single underlying I/O call instead of looping until the request is fully satisfied. They are used internally by classes likemachine.UARTto implement their ownread/write– but no standard stream class binds them as Python-callableread1/readinto1/write1methods.
- class io.StringIO(string: str = '')¶
In-memory file-like object for text-mode input/output (similar to a normal file opened with the “t” modifier). Initial contents can be specified with the string parameter (which should be a normal string). Instances also support the context-manager protocol (usable in a
withstatement).- read(size: int = -1) str¶
Read and return up to size characters. If size is omitted or negative, read and return all remaining contents.
- readline(size: int = -1) str¶
Read and return one line. If size is given, at most size characters are read.
- readinto(buf: bytearray) int¶
Read into the pre-allocated, writable buffer buf and return the number of bytes read.
- seek(offset: int, whence: int = 0) int¶
Change the stream position to offset relative to whence (
0= start,1= current,2= end) and return the new absolute position.
- close() None¶
Close the stream and free the underlying buffer. Further operations on a closed stream raise
ValueError.
- class io.StringIO(alloc_size: int)
Create an empty
StringIOobject preallocated to hold up to alloc_size bytes, so writing up to that many bytes will not reallocate the buffer (avoiding an out-of-memory situation or memory fragmentation). This constructor is a MicroPython extension recommended only for special cases and system-level libraries, not for end-user applications.Difference to CPython
This constructor is a MicroPython extension.
- read(size: int = -1) str
Read and return up to size characters. If size is omitted or negative, read and return all remaining contents.
- readline(size: int = -1) str
Read and return one line. If size is given, at most size characters are read.
- readinto(buf: bytearray) int
Read into the pre-allocated, writable buffer buf and return the number of bytes read.
- seek(offset: int, whence: int = 0) int
Change the stream position to offset relative to whence (
0= start,1= current,2= end) and return the new absolute position.
- tell() int
Return the current stream position.
- flush() None
Flush the write buffers. This is a no-op for an in-memory stream.
- close() None
Close the stream and free the underlying buffer. Further operations on a closed stream raise
ValueError.
- getvalue() str
Return the current contents of the underlying buffer.
- class io.BytesIO(string: bytes = b'')¶
In-memory file-like object for binary-mode input/output (similar to a normal file opened with the “b” modifier). Initial contents can be specified with the string parameter (which should be a bytes object). Instances also support the context-manager protocol (usable in a
withstatement).- read(size: int = -1) bytes¶
Read and return up to size bytes. If size is omitted or negative, read and return all remaining contents.
- readline(size: int = -1) bytes¶
Read and return one line. If size is given, at most size bytes are read.
- readinto(buf: bytearray) int¶
Read into the pre-allocated, writable buffer buf and return the number of bytes read.
- seek(offset: int, whence: int = 0) int¶
Change the stream position to offset relative to whence (
0= start,1= current,2= end) and return the new absolute position.
- close() None¶
Close the stream and free the underlying buffer. Further operations on a closed stream raise
ValueError.
- class io.BytesIO(alloc_size: int)
Create an empty
BytesIOobject preallocated to hold up to alloc_size bytes, so writing up to that many bytes will not reallocate the buffer (avoiding an out-of-memory situation or memory fragmentation). This constructor is a MicroPython extension recommended only for special cases and system-level libraries, not for end-user applications.Difference to CPython
This constructor is a MicroPython extension.
- read(size: int = -1) bytes
Read and return up to size bytes. If size is omitted or negative, read and return all remaining contents.
- readline(size: int = -1) bytes
Read and return one line. If size is given, at most size bytes are read.
- readinto(buf: bytearray) int
Read into the pre-allocated, writable buffer buf and return the number of bytes read.
- seek(offset: int, whence: int = 0) int
Change the stream position to offset relative to whence (
0= start,1= current,2= end) and return the new absolute position.
- tell() int
Return the current stream position.
- flush() None
Flush the write buffers. This is a no-op for an in-memory stream.
- close() None
Close the stream and free the underlying buffer. Further operations on a closed stream raise
ValueError.
- getvalue() bytes
Return the current contents of the underlying buffer.