2019-09-03 21:08:51 -06:00
|
|
|
## Nim-LibP2P
|
2019-09-24 11:48:23 -06:00
|
|
|
## Copyright (c) 2019 Status Research & Development GmbH
|
2019-09-03 21:08:51 -06:00
|
|
|
## Licensed under either of
|
|
|
|
## * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
|
|
## * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
|
|
## at your option.
|
|
|
|
## This file may not be copied, modified, or distributed except according to
|
|
|
|
## those terms.
|
|
|
|
|
2020-05-19 18:14:15 -06:00
|
|
|
import oids, deques
|
2020-06-19 11:29:43 -06:00
|
|
|
import chronos, chronicles, metrics
|
2019-09-12 11:07:34 -06:00
|
|
|
import types,
|
2019-09-08 01:59:14 -06:00
|
|
|
coder,
|
2020-07-17 12:44:41 -06:00
|
|
|
../muxer,
|
2019-09-08 01:59:14 -06:00
|
|
|
nimcrypto/utils,
|
2020-06-19 11:29:43 -06:00
|
|
|
../../stream/connection,
|
2019-09-12 11:07:34 -06:00
|
|
|
../../stream/bufferstream,
|
2020-06-29 09:15:31 -06:00
|
|
|
../../peerinfo
|
2019-09-03 21:08:51 -06:00
|
|
|
|
2020-06-19 11:29:43 -06:00
|
|
|
export connection
|
2020-05-06 18:31:47 +02:00
|
|
|
|
2019-09-09 20:15:52 -06:00
|
|
|
logScope:
|
2020-06-10 11:48:01 +03:00
|
|
|
topics = "mplexchannel"
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2020-05-19 18:14:15 -06:00
|
|
|
## Channel half-closed states
|
|
|
|
##
|
|
|
|
## | State | Closed local | Closed remote
|
|
|
|
## |=============================================
|
|
|
|
## | Read | Yes (until EOF) | No
|
|
|
|
## | Write | No | Yes
|
|
|
|
##
|
|
|
|
|
2020-05-29 10:24:38 -06:00
|
|
|
# TODO: this is one place where we need to use
|
|
|
|
# a proper state machine, but I've opted out of
|
|
|
|
# it for now for two reasons:
|
|
|
|
#
|
|
|
|
# 1) we don't have that many states to manage
|
|
|
|
# 2) I'm not sure if adding the state machine
|
|
|
|
# would have simplified or complicated the code
|
|
|
|
#
|
|
|
|
# But now that this is in place, we should perhaps
|
|
|
|
# reconsider reworking it again, this time with a
|
|
|
|
# more formal approach.
|
|
|
|
#
|
|
|
|
|
2019-09-03 21:08:51 -06:00
|
|
|
type
|
2019-09-12 11:07:34 -06:00
|
|
|
LPChannel* = ref object of BufferStream
|
2020-05-19 18:14:15 -06:00
|
|
|
id*: uint64 # channel id
|
|
|
|
name*: string # name of the channel (for debugging)
|
|
|
|
conn*: Connection # wrapped connection used to for writing
|
|
|
|
initiator*: bool # initiated remotely or locally flag
|
|
|
|
isLazy*: bool # is channel lazy
|
2020-05-29 10:24:38 -06:00
|
|
|
isOpen*: bool # has channel been opened (only used with isLazy)
|
2020-05-19 18:14:15 -06:00
|
|
|
closedLocal*: bool # has channel been closed locally
|
|
|
|
msgCode*: MessageType # cached in/out message code
|
|
|
|
closeCode*: MessageType # cached in/out close code
|
|
|
|
resetCode*: MessageType # cached in/out reset code
|
|
|
|
|
|
|
|
proc open*(s: LPChannel) {.async, gcsafe.}
|
|
|
|
|
|
|
|
template withWriteLock(lock: AsyncLock, body: untyped): untyped =
|
|
|
|
try:
|
|
|
|
await lock.acquire()
|
|
|
|
body
|
|
|
|
finally:
|
2020-06-11 08:45:59 -06:00
|
|
|
if not(isNil(lock)) and lock.locked:
|
|
|
|
lock.release()
|
2020-05-19 18:14:15 -06:00
|
|
|
|
|
|
|
proc closeMessage(s: LPChannel) {.async.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
2020-06-01 17:19:53 -06:00
|
|
|
## send close message - this will not raise
|
|
|
|
## on EOF or Closed
|
2020-06-29 09:15:31 -06:00
|
|
|
withWriteLock(s.writeLock):
|
|
|
|
trace "sending close message"
|
2019-09-07 17:32:32 -06:00
|
|
|
|
2020-06-29 09:15:31 -06:00
|
|
|
await s.conn.writeMsg(s.id, s.closeCode) # write close
|
2020-03-27 08:25:52 -06:00
|
|
|
|
2020-05-19 18:14:15 -06:00
|
|
|
proc resetMessage(s: LPChannel) {.async.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
2020-06-01 17:19:53 -06:00
|
|
|
## send reset message - this will not raise
|
2020-09-04 19:30:45 +03:00
|
|
|
try:
|
2020-05-19 18:14:15 -06:00
|
|
|
withWriteLock(s.writeLock):
|
2020-06-29 09:15:31 -06:00
|
|
|
trace "sending reset message"
|
2020-05-19 18:14:15 -06:00
|
|
|
await s.conn.writeMsg(s.id, s.resetCode) # write reset
|
2020-09-04 19:30:45 +03:00
|
|
|
except CancelledError:
|
|
|
|
# This procedure is called from one place and never awaited, so there no
|
|
|
|
# need to re-raise CancelledError.
|
|
|
|
trace "Unexpected cancellation while resetting channel"
|
|
|
|
except LPStreamEOFError as exc:
|
|
|
|
trace "muxed connection EOF", exc = exc.msg
|
|
|
|
except LPStreamClosedError as exc:
|
|
|
|
trace "muxed connection closed", exc = exc.msg
|
|
|
|
except LPStreamIncompleteError as exc:
|
|
|
|
trace "incomplete message", exc = exc.msg
|
|
|
|
except CatchableError as exc:
|
|
|
|
trace "Unhandled exception leak", exc = exc.msg
|
2020-05-19 18:14:15 -06:00
|
|
|
|
|
|
|
proc open*(s: LPChannel) {.async, gcsafe.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
2020-05-19 18:14:15 -06:00
|
|
|
## NOTE: Don't call withExcAndLock or withWriteLock,
|
|
|
|
## because this already gets called from writeHandler
|
|
|
|
## which is locked
|
2020-06-29 09:15:31 -06:00
|
|
|
await s.conn.writeMsg(s.id, MessageType.New, s.name)
|
|
|
|
trace "opened channel"
|
|
|
|
s.isOpen = true
|
2020-05-19 18:14:15 -06:00
|
|
|
|
|
|
|
proc closeRemote*(s: LPChannel) {.async.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
|
|
|
trace "got EOF, closing channel"
|
2020-07-17 12:44:41 -06:00
|
|
|
try:
|
|
|
|
await s.drainBuffer()
|
|
|
|
s.isEof = true # set EOF immediately to prevent further reads
|
2020-08-10 16:17:11 -06:00
|
|
|
# close parent bufferstream to prevent further reads
|
|
|
|
await procCall BufferStream(s).close()
|
2020-07-18 11:00:44 -06:00
|
|
|
|
2020-07-17 12:44:41 -06:00
|
|
|
trace "channel closed on EOF"
|
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
|
|
|
except CatchableError as exc:
|
|
|
|
trace "exception closing remote channel", exc = exc.msg
|
2020-03-27 08:25:52 -06:00
|
|
|
|
2020-05-19 18:14:15 -06:00
|
|
|
method closed*(s: LPChannel): bool =
|
|
|
|
## this emulates half-closed behavior
|
|
|
|
## when closed locally writing is
|
2020-05-29 10:24:38 -06:00
|
|
|
## disabled - see the table in the
|
2020-05-19 18:14:15 -06:00
|
|
|
## header of the file
|
|
|
|
s.closedLocal
|
2019-12-03 22:44:54 -06:00
|
|
|
|
2020-05-23 10:50:05 -06:00
|
|
|
method reset*(s: LPChannel) {.base, async, gcsafe.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
|
|
|
if s.closedLocal and s.isEof:
|
|
|
|
trace "channel already closed or reset"
|
|
|
|
return
|
|
|
|
|
2020-07-27 13:33:51 -06:00
|
|
|
trace "resetting channel"
|
|
|
|
|
2020-09-04 19:30:45 +03:00
|
|
|
asyncSpawn s.resetMessage()
|
2020-07-12 10:37:10 -06:00
|
|
|
|
2020-07-17 12:44:41 -06:00
|
|
|
try:
|
|
|
|
# drain the buffer before closing
|
|
|
|
await s.drainBuffer()
|
|
|
|
await procCall BufferStream(s).close()
|
2020-07-12 10:37:10 -06:00
|
|
|
|
2020-07-17 12:44:41 -06:00
|
|
|
s.isEof = true
|
|
|
|
s.closedLocal = true
|
|
|
|
|
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
|
|
|
except CatchableError as exc:
|
|
|
|
trace "exception in reset", exc = exc.msg
|
2020-05-23 10:50:05 -06:00
|
|
|
|
2020-06-29 09:15:31 -06:00
|
|
|
trace "channel reset"
|
|
|
|
|
2020-05-23 10:50:05 -06:00
|
|
|
method close*(s: LPChannel) {.async, gcsafe.} =
|
2020-06-29 09:15:31 -06:00
|
|
|
logScope:
|
|
|
|
id = s.id
|
|
|
|
initiator = s.initiator
|
|
|
|
name = s.name
|
|
|
|
oid = $s.oid
|
|
|
|
peer = $s.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
2020-05-23 10:50:05 -06:00
|
|
|
if s.closedLocal:
|
2020-06-29 09:15:31 -06:00
|
|
|
trace "channel already closed"
|
2020-05-23 10:50:05 -06:00
|
|
|
return
|
|
|
|
|
2020-06-29 09:15:31 -06:00
|
|
|
trace "closing local lpchannel"
|
|
|
|
|
|
|
|
proc closeInternal() {.async.} =
|
2020-05-23 10:50:05 -06:00
|
|
|
try:
|
|
|
|
await s.closeMessage().wait(2.minutes)
|
2020-05-29 10:24:38 -06:00
|
|
|
if s.atEof: # already closed by remote close parent buffer immediately
|
2020-05-23 10:50:05 -06:00
|
|
|
await procCall BufferStream(s).close()
|
2020-09-04 19:30:45 +03:00
|
|
|
except CancelledError:
|
|
|
|
trace "Unexpected cancellation while closing channel"
|
2020-07-12 10:37:10 -06:00
|
|
|
await s.reset()
|
2020-09-04 19:30:45 +03:00
|
|
|
# This is top-level procedure which will work as separate task, so it
|
|
|
|
# do not need to propogate CancelledError.
|
2020-05-23 10:50:05 -06:00
|
|
|
except CatchableError as exc:
|
2020-07-27 13:33:51 -06:00
|
|
|
trace "exception closing channel", exc = exc.msg
|
2020-07-12 10:37:10 -06:00
|
|
|
await s.reset()
|
2020-05-23 10:50:05 -06:00
|
|
|
|
2020-06-29 09:15:31 -06:00
|
|
|
trace "lpchannel closed local"
|
2020-05-23 10:50:05 -06:00
|
|
|
|
2020-06-19 11:29:43 -06:00
|
|
|
s.closedLocal = true
|
2020-09-04 19:30:45 +03:00
|
|
|
# All the errors are handled inside `closeInternal()` procedure.
|
|
|
|
asyncSpawn closeInternal()
|
2020-07-17 12:44:41 -06:00
|
|
|
|
|
|
|
method initStream*(s: LPChannel) =
|
|
|
|
if s.objName.len == 0:
|
|
|
|
s.objName = "LPChannel"
|
|
|
|
|
2020-08-04 07:22:05 -06:00
|
|
|
s.timeoutHandler = proc() {.async, gcsafe.} =
|
|
|
|
trace "idle timeout expired, resetting LPChannel"
|
|
|
|
await s.reset()
|
2020-07-17 12:44:41 -06:00
|
|
|
|
2020-08-04 07:22:05 -06:00
|
|
|
procCall BufferStream(s).initStream()
|
2020-07-17 12:44:41 -06:00
|
|
|
|
|
|
|
proc init*(
|
|
|
|
L: type LPChannel,
|
|
|
|
id: uint64,
|
|
|
|
conn: Connection,
|
|
|
|
initiator: bool,
|
|
|
|
name: string = "",
|
|
|
|
size: int = DefaultBufferSize,
|
|
|
|
lazy: bool = false,
|
|
|
|
timeout: Duration = DefaultChanTimeout): LPChannel =
|
|
|
|
|
|
|
|
let chann = L(
|
|
|
|
id: id,
|
|
|
|
name: name,
|
|
|
|
conn: conn,
|
|
|
|
initiator: initiator,
|
|
|
|
isLazy: lazy,
|
|
|
|
timeout: timeout,
|
|
|
|
msgCode: if initiator: MessageType.MsgOut else: MessageType.MsgIn,
|
|
|
|
closeCode: if initiator: MessageType.CloseOut else: MessageType.CloseIn,
|
|
|
|
resetCode: if initiator: MessageType.ResetOut else: MessageType.ResetIn,
|
|
|
|
dir: if initiator: Direction.Out else: Direction.In)
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
id = chann.id
|
|
|
|
initiator = chann.initiator
|
|
|
|
name = chann.name
|
|
|
|
oid = $chann.oid
|
|
|
|
peer = $chann.conn.peerInfo
|
|
|
|
# stack = getStackTrace()
|
|
|
|
|
|
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
try:
|
|
|
|
if chann.isLazy and not(chann.isOpen):
|
|
|
|
await chann.open()
|
|
|
|
|
|
|
|
# writes should happen in sequence
|
2020-08-15 07:58:30 +02:00
|
|
|
trace "sending data", len = data.len
|
2020-07-17 12:44:41 -06:00
|
|
|
|
|
|
|
await conn.writeMsg(chann.id,
|
|
|
|
chann.msgCode,
|
|
|
|
data)
|
|
|
|
except CatchableError as exc:
|
|
|
|
trace "exception in lpchannel write handler", exc = exc.msg
|
|
|
|
await chann.reset()
|
|
|
|
raise exc
|
|
|
|
|
|
|
|
chann.initBufferStream(writeHandler, size)
|
|
|
|
when chronicles.enabledLogLevel == LogLevel.TRACE:
|
|
|
|
chann.name = if chann.name.len > 0: chann.name else: $chann.oid
|
|
|
|
|
|
|
|
trace "created new lpchannel"
|
|
|
|
|
|
|
|
return chann
|