mplex stream close reset done

This commit is contained in:
zixuanzh
2018-11-20 21:46:18 -05:00
parent e047752d82
commit d83b6ef459
2 changed files with 36 additions and 28 deletions

View File

@ -19,6 +19,7 @@ class Mplex(IMuxedConn):
self.buffers = {} self.buffers = {}
self.streams = {} self.streams = {}
self.stream_queue = asyncio.Queue() self.stream_queue = asyncio.Queue()
self.conn_lock = asyncio.Lock()
# The initiator need not read upon construction time. # The initiator need not read upon construction time.
# It should read when the user decides that it wants to read from the constructed stream. # It should read when the user decides that it wants to read from the constructed stream.

View File

@ -1,3 +1,4 @@
import asyncio
from .constants import HEADER_TAGS from .constants import HEADER_TAGS
from ..muxed_stream_interface import IMuxedStream from ..muxed_stream_interface import IMuxedStream
@ -7,22 +8,21 @@ class MplexStream(IMuxedStream):
reference: https://github.com/libp2p/go-mplex/blob/master/stream.go reference: https://github.com/libp2p/go-mplex/blob/master/stream.go
""" """
def __init__(self, stream_id, initiator, muxed_conn): def __init__(self, stream_id, initiator, mplex_conn):
""" """
create new MuxedStream in muxer create new MuxedStream in muxer
:param stream_id: stream stream id :param stream_id: stream stream id
:param initiator: boolean if this is an initiator :param initiator: boolean if this is an initiator
:param muxed_conn: muxed connection of this muxed_stream :param mplex_conn: muxed connection of this muxed_stream
""" """
self.stream_id = stream_id self.stream_id = stream_id
self.initiator = initiator self.initiator = initiator
self.muxed_conn = muxed_conn self.mplex_conn = mplex_conn
self.read_deadline = None self.read_deadline = None
self.write_deadline = None self.write_deadline = None
self.local_closed = False self.local_closed = False
self.remote_closed = False self.remote_closed = False
self.stream_lock = asyncio.Lock()
def get_flag(self, action): def get_flag(self, action):
""" """
@ -40,50 +40,57 @@ class MplexStream(IMuxedStream):
read messages associated with stream from buffer til end of file read messages associated with stream from buffer til end of file
:return: bytes of input :return: bytes of input
""" """
return await self.muxed_conn.read_buffer(self.stream_id) return await self.mplex_conn.read_buffer(self.stream_id)
async def write(self, data): async def write(self, data):
""" """
write to stream write to stream
:return: number of bytes written :return: number of bytes written
""" """
return await self.muxed_conn.send_message(self.get_flag("MESSAGE"), data, self.stream_id) return await self.mplex_conn.send_message(self.get_flag("MESSAGE"), data, self.stream_id)
async def close(self): async def close(self):
""" """
close stream Closing a stream closes it for writing and closes the remote end for reading
but allows writing in the other direction.
:return: true if successful :return: true if successful
""" """
#TODO error handling with timeout
await self.mplex_conn.send_message(self.get_flag("CLOSE"), None, self.stream_id)
if self.local_closed and self.remote_closed: remote_lock = ""
return True async with self.stream_lock:
if self.local_closed:
return True
self.local_closed = True
remote_lock = self.remote_closed
await self.muxed_conn.send_message(self.get_flag("CLOSE"), None, self.stream_id) if remote_lock:
self.muxed_conn.streams.pop(self.stream_id) async with self.mplex_conn.conn_lock:
self.mplex_conn.streams.pop(self.stream_id)
self.local_closed = True
self.remote_closed = True
return True return True
def reset(self): async def reset(self):
""" """
closes both ends of the stream closes both ends of the stream
tells this remote side to hang up tells this remote side to hang up
:return: true if successful :return: true if successful
""" """
# TODO behavior not fully understood async with self.stream_lock:
pass if self.remote_closed and self.local_closed:
# if self.local_closed and self.remote_closed: return True
# return True
# if not self.remote_closed:
# self.muxed_conn.send_message(self.get_flag("RESET"), None, self.id) await self.mplex_conn.send_message(self.get_flag("RESET"), None, self.stream_id)
# self.muxed_conn.streams.pop(self.id, None)
# self.local_closed = True
# self.local_closed = True self.remote_closed = True
# self.remote_closed = True
# async with self.mplex_conn.conn_lock:
# return True self.mplex_conn.streams.pop(self.stream_id, None)
return True
# TODO deadline not in use # TODO deadline not in use
def set_deadline(self, ttl): def set_deadline(self, ttl):