Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

LibP2P network backends #278

Merged
merged 18 commits into from
Jun 24, 2019
Merged

LibP2P network backends #278

merged 18 commits into from
Jun 24, 2019

Conversation

zah
Copy link
Contributor

@zah zah commented Jun 5, 2019

You can switch between the available backends by compiling with -d:network_type=<backend_name>.

The default backend is now "libp2p_spec" which implements the spec published here:
https://github.com/ethereum/eth2.0-specs/tree/dev/specs/networking

A forward-looking backend called "libp2p_native" attempts to take advantage of the existing multi-plexing capabilities of LibP2P. Hopefully, we'll be able to switch to it once Multistream-2.0 lands and eliminates the currently present overhead when using a separate stream for each request.

The old RLPx backend is still available when compiling with -d:network_type=rlpx.

@@ -166,68 +136,55 @@ proc sendBytes(stream: P2PStream, bytes: Bytes) {.async.} =

proc makeEth2Request(peer: Peer, protocolId: string, requestBytes: Bytes,
ResponseMsg: type,
timeout = 10000): Future[Option[ResponseMsg]] {.async.} =
timeout = 10.seconds): Future[Option[ResponseMsg]] {.async.} =
var stream = await peer.network.daemon.openStream(peer.id, @[protocolId])
# TODO how does openStream fail? Set a timeout here and handle it
let sent = await stream.transp.write(requestBytes)
# TODO: Should I check that `sent` is equal to the desired number of bytes
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should check and if result != len(requestBytes), then your connection is dead, but still requires transport.close().

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If performing these steps is necessary every single time, why don't we introduce a simpler API where close is being called automatically on failure?

Are there situations where it won't be appropriate to call close immediately?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Its actually very bad idea to make this simple API, because its very easy to make race condition or double close in such way. And when we fix rlpx.nim with @kdeme you will see that there will be just one close call in code.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you provide an example of these race conditions, so I can understand the problem better?

Isn't it possible to make the API easier to use in general if double close wasn't a problem in the first place?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In rlpx, the "transport already closed" exceptions do not happen because of a double close.
In fact, from what I understand, a double close will not give trouble and the API takes care of it: see https://github.com/status-im/nim-chronos/blob/a8a1138b8b66ac51378897ef126b3783dcf9a2ca/chronos/transports/stream.nim#L1750

In rlpx, the "transport already closed" exceptions happen when trying to read or write data on a transport which we already closed. This can (and niw is ?mostly?) being taken care of by just handling the exceptions. But it feels a bit unnecessary and expensive. Probably also less error prone if the read/writes just don't occur any more. I believe this is what @cheatfate means with the races.

include eth/p2p/p2p_tracing

import typetraits

proc readMsg(stream: P2PStream, MsgType: type,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This procedure do not have any error handlers.

proc readMsg(stream: P2PStream, MsgType: type,
timeout = 10000): Future[Option[MsgType]] {.async.} =
timeout = 10.seconds): Future[Option[MsgType]] {.async.} =
Copy link
Contributor

@cheatfate cheatfate Jun 6, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Timeout here is only used when prefix is received and not used for receiving body.

@zah zah force-pushed the libp2p-builds branch from d3bc777 to 50d3ca2 Compare June 12, 2019 12:47
beacon_chain/beacon_node.nim Show resolved Hide resolved
beacon_chain/conf.nim Show resolved Hide resolved
beacon_chain/eth2_network.nim Show resolved Hide resolved
@zah zah force-pushed the libp2p-builds branch 2 times, most recently from ba0c65f to 0bc8e01 Compare June 22, 2019 10:34
@zah zah force-pushed the libp2p-builds branch from 812a1bf to c751285 Compare June 24, 2019 02:36
@zah zah merged commit c751285 into master Jun 24, 2019
@delete-merged-branch delete-merged-branch bot deleted the libp2p-builds branch June 24, 2019 02:41
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants