|
|
- package p2p
-
- import (
- "context"
- "fmt"
- "sync"
-
- "github.com/gogo/protobuf/proto"
- "github.com/tendermint/tendermint/types"
- )
-
- // Envelope contains a message with sender/receiver routing info.
- type Envelope struct {
- From types.NodeID // sender (empty if outbound)
- To types.NodeID // receiver (empty if inbound)
- Broadcast bool // send to all connected peers (ignores To)
- Message proto.Message // message payload
-
- // channelID is for internal Router use, set on outbound messages to inform
- // the sendPeer() goroutine which transport channel to use.
- //
- // FIXME: If we migrate the Transport API to a byte-oriented multi-stream
- // API, this will no longer be necessary since each channel will be mapped
- // onto a stream during channel/peer setup. See:
- // https://github.com/tendermint/spec/pull/227
- channelID ChannelID
- }
-
- // Wrapper is a Protobuf message that can contain a variety of inner messages
- // (e.g. via oneof fields). If a Channel's message type implements Wrapper, the
- // Router will automatically wrap outbound messages and unwrap inbound messages,
- // such that reactors do not have to do this themselves.
- type Wrapper interface {
- proto.Message
-
- // Wrap will take a message and wrap it in this one if possible.
- Wrap(proto.Message) error
-
- // Unwrap will unwrap the inner message contained in this message.
- Unwrap() (proto.Message, error)
- }
-
- // PeerError is a peer error reported via Channel.Error.
- //
- // FIXME: This currently just disconnects the peer, which is too simplistic.
- // For example, some errors should be logged, some should cause disconnects,
- // and some should ban the peer.
- //
- // FIXME: This should probably be replaced by a more general PeerBehavior
- // concept that can mark good and bad behavior and contributes to peer scoring.
- // It should possibly also allow reactors to request explicit actions, e.g.
- // disconnection or banning, in addition to doing this based on aggregates.
- type PeerError struct {
- NodeID types.NodeID
- Err error
- }
-
- func (pe PeerError) Error() string { return fmt.Sprintf("peer=%q: %s", pe.NodeID, pe.Err.Error()) }
- func (pe PeerError) Unwrap() error { return pe.Err }
-
- // Channel is a bidirectional channel to exchange Protobuf messages with peers.
- // Each message is wrapped in an Envelope to specify its sender and receiver.
- type Channel struct {
- ID ChannelID
- In <-chan Envelope // inbound messages (peers to reactors)
- Out chan<- Envelope // outbound messages (reactors to peers)
- errCh chan<- PeerError // peer error reporting
-
- messageType proto.Message // the channel's message type, used for unmarshaling
- }
-
- // NewChannel creates a new channel. It is primarily for internal and test
- // use, reactors should use Router.OpenChannel().
- func NewChannel(
- id ChannelID,
- messageType proto.Message,
- inCh <-chan Envelope,
- outCh chan<- Envelope,
- errCh chan<- PeerError,
- ) *Channel {
- return &Channel{
- ID: id,
- messageType: messageType,
- In: inCh,
- Out: outCh,
- errCh: errCh,
- }
- }
-
- // Send blocks until the envelope has been sent, or until ctx ends.
- // An error only occurs if the context ends before the send completes.
- func (ch *Channel) Send(ctx context.Context, envelope Envelope) error {
- select {
- case <-ctx.Done():
- return ctx.Err()
- case ch.Out <- envelope:
- return nil
- }
- }
-
- // SendError blocks until the given error has been sent, or ctx ends.
- // An error only occurs if the context ends before the send completes.
- func (ch *Channel) SendError(ctx context.Context, pe PeerError) error {
- select {
- case <-ctx.Done():
- return ctx.Err()
- case ch.errCh <- pe:
- return nil
- }
- }
-
- // Receive returns a new unbuffered iterator to receive messages from ch.
- // The iterator runs until ctx ends.
- func (ch *Channel) Receive(ctx context.Context) *ChannelIterator {
- iter := &ChannelIterator{
- pipe: make(chan Envelope), // unbuffered
- }
- go func() {
- defer close(iter.pipe)
- iteratorWorker(ctx, ch, iter.pipe)
- }()
- return iter
- }
-
- // ChannelIterator provides a context-aware path for callers
- // (reactors) to process messages from the P2P layer without relying
- // on the implementation details of the P2P layer. Channel provides
- // access to it's Outbound stream as an iterator, and the
- // MergedChannelIterator makes it possible to combine multiple
- // channels into a single iterator.
- type ChannelIterator struct {
- pipe chan Envelope
- current *Envelope
- }
-
- func iteratorWorker(ctx context.Context, ch *Channel, pipe chan Envelope) {
- for {
- select {
- case <-ctx.Done():
- return
- case envelope := <-ch.In:
- select {
- case <-ctx.Done():
- return
- case pipe <- envelope:
- }
- }
- }
- }
-
- // Next returns true when the Envelope value has advanced, and false
- // when the context is canceled or iteration should stop. If an iterator has returned false,
- // it will never return true again.
- // in general, use Next, as in:
- //
- // for iter.Next(ctx) {
- // envelope := iter.Envelope()
- // // ... do things ...
- // }
- //
- func (iter *ChannelIterator) Next(ctx context.Context) bool {
- select {
- case <-ctx.Done():
- iter.current = nil
- return false
- case envelope, ok := <-iter.pipe:
- if !ok {
- iter.current = nil
- return false
- }
-
- iter.current = &envelope
-
- return true
- }
- }
-
- // Envelope returns the current Envelope object held by the
- // iterator. When the last call to Next returned true, Envelope will
- // return a non-nil object. If Next returned false then Envelope is
- // always nil.
- func (iter *ChannelIterator) Envelope() *Envelope { return iter.current }
-
- // MergedChannelIterator produces an iterator that merges the
- // messages from the given channels in arbitrary order.
- //
- // This allows the caller to consume messages from multiple channels
- // without needing to manage the concurrency separately.
- func MergedChannelIterator(ctx context.Context, chs ...*Channel) *ChannelIterator {
- iter := &ChannelIterator{
- pipe: make(chan Envelope), // unbuffered
- }
- wg := new(sync.WaitGroup)
-
- done := make(chan struct{})
- go func() { defer close(done); wg.Wait() }()
-
- go func() {
- defer close(iter.pipe)
- // we could return early if the context is canceled,
- // but this is safer because it means the pipe stays
- // open until all of the ch worker threads end, which
- // should happen very quickly.
- <-done
- }()
-
- for _, ch := range chs {
- wg.Add(1)
- go func(ch *Channel) {
- defer wg.Done()
- iteratorWorker(ctx, ch, iter.pipe)
- }(ch)
- }
-
- return iter
- }
|