Skip to content

Commit 9e7db99

Browse files
committed
micropython/streampair: Package to create bi-directional linked stream objects.
Signed-off-by: Andrew Leech <[email protected]>
1 parent 60d1370 commit 9e7db99

File tree

3 files changed

+142
-0
lines changed

3 files changed

+142
-0
lines changed

micropython/streampair/manifest.py

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
metadata(
2+
description="Create a bi-directional linked pair of stream objects", version="0.0.1"
3+
)
4+
5+
module("streampair.py")

micropython/streampair/streampair.py

+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
import io
2+
3+
from collections import deque
4+
from micropython import ringbuffer, const
5+
6+
try:
7+
from typing import Union, Tuple
8+
except:
9+
pass
10+
11+
# From micropython/py/stream.h
12+
_MP_STREAM_ERROR = const(-1)
13+
_MP_STREAM_FLUSH = const(1)
14+
_MP_STREAM_SEEK = const(2)
15+
_MP_STREAM_POLL = const(3)
16+
_MP_STREAM_CLOSE = const(4)
17+
_MP_STREAM_POLL_RD = const(0x0001)
18+
19+
20+
def streampair(buffer_size: Union[int, Tuple[int, int]]=256):
21+
"""
22+
Returns two bi-directional linked stream objects where writes to one can be read from the other and vice/versa.
23+
This can be used somewhat similarly to a socket.socketpair in python, like a pipe
24+
of data that can be used to connect stream consumers (eg. asyncio.StreamWriter, mock Uart)
25+
"""
26+
try:
27+
size_a, size_b = buffer_size
28+
except TypeError:
29+
size_a = size_b = buffer_size
30+
31+
a = ringbuffer(size_a)
32+
b = ringbuffer(size_b)
33+
return StreamPair(a, b), StreamPair(b, a)
34+
35+
36+
class StreamPair(io.IOBase):
37+
38+
def __init__(self, own: ringbuffer, other: ringbuffer):
39+
self.own = own
40+
self.other = other
41+
super().__init__()
42+
43+
def read(self, nbytes=-1):
44+
return self.own.read(nbytes)
45+
46+
def readline(self):
47+
return self.own.readline()
48+
49+
def readinto(self, buf, limit=-1):
50+
return self.own.readinto(buf, limit)
51+
52+
def write(self, data):
53+
return self.other.write(data)
54+
55+
def seek(self, offset, whence):
56+
return self.own.seek(offset, whence)
57+
58+
def flush(self):
59+
self.own.flush()
60+
self.other.flush()
61+
62+
def close(self):
63+
self.own.close()
64+
self.other.close()
65+
66+
def any(self):
67+
return self.own.any()
68+
69+
def ioctl(self, op, arg):
70+
if op == _MP_STREAM_POLL:
71+
if self.any():
72+
return _MP_STREAM_POLL_RD
73+
return 0
74+
75+
elif op ==_MP_STREAM_FLUSH:
76+
return self.flush()
77+
elif op ==_MP_STREAM_SEEK:
78+
return self.seek(arg[0], arg[1])
79+
elif op ==_MP_STREAM_CLOSE:
80+
return self.close()
81+
82+
else:
83+
return _MP_STREAM_ERROR
+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import asyncio
2+
import unittest
3+
from streampair import streampair
4+
5+
def async_test(f):
6+
"""
7+
Decorator to run an async test function
8+
"""
9+
def wrapper(*args, **kwargs):
10+
loop = asyncio.new_event_loop()
11+
# loop.set_exception_handler(_exception_handler)
12+
t = loop.create_task(f(*args, **kwargs))
13+
loop.run_until_complete(t)
14+
15+
return wrapper
16+
17+
class StreamPairTestCase(unittest.TestCase):
18+
19+
def test_streampair(self):
20+
a, b = streampair()
21+
assert a.write(b"foo") == 3
22+
assert b.write(b"bar") == 3
23+
24+
assert (r := a.read()) == b"bar", r
25+
assert (r := b.read()) == b"foo", r
26+
27+
@async_test
28+
async def test_async_streampair(self):
29+
a, b = streampair()
30+
ar = asyncio.StreamReader(a)
31+
bw = asyncio.StreamWriter(b)
32+
33+
br = asyncio.StreamReader(b)
34+
aw = asyncio.StreamWriter(a)
35+
36+
aw.write(b"foo\n")
37+
await aw.drain()
38+
assert not a.any()
39+
assert b.any()
40+
assert (r := await br.readline()) == b"foo\n", r
41+
assert not b.any()
42+
assert not a.any()
43+
44+
bw.write(b"bar\n")
45+
await bw.drain()
46+
assert not b.any()
47+
assert a.any()
48+
assert (r := await ar.readline()) == b"bar\n", r
49+
assert not b.any()
50+
assert not a.any()
51+
52+
53+
if __name__ == "__main__":
54+
unittest.main()

0 commit comments

Comments
 (0)