2018-07-04 10:51:47 +00:00
|
|
|
package swarm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/mux"
|
|
|
|
"github.com/libp2p/go-libp2p-core/network"
|
|
|
|
"github.com/libp2p/go-libp2p-core/protocol"
|
2018-07-04 10:51:47 +00:00
|
|
|
)
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
// Validate Stream conforms to the go-libp2p-net Stream interface
|
|
|
|
var _ network.Stream = &Stream{}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
// Stream is the stream type used by swarm. In general, you won't use this type
|
|
|
|
// directly.
|
|
|
|
type Stream struct {
|
2021-06-16 20:19:45 +00:00
|
|
|
id uint32
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
stream mux.MuxedStream
|
2018-07-04 10:51:47 +00:00
|
|
|
conn *Conn
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
closeOnce sync.Once
|
2018-07-04 10:51:47 +00:00
|
|
|
|
|
|
|
notifyLk sync.Mutex
|
|
|
|
|
|
|
|
protocol atomic.Value
|
2019-06-09 07:24:20 +00:00
|
|
|
|
|
|
|
stat network.Stat
|
2018-07-04 10:51:47 +00:00
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
func (s *Stream) ID() string {
|
|
|
|
// format: <first 10 chars of peer id>-<global conn ordinal>-<global stream ordinal>
|
|
|
|
return fmt.Sprintf("%s-%d", s.conn.ID(), s.id)
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
func (s *Stream) String() string {
|
|
|
|
return fmt.Sprintf(
|
|
|
|
"<swarm.Stream[%s] %s (%s) <-> %s (%s)>",
|
|
|
|
s.conn.conn.Transport(),
|
|
|
|
s.conn.LocalMultiaddr(),
|
|
|
|
s.conn.LocalPeer(),
|
|
|
|
s.conn.RemoteMultiaddr(),
|
|
|
|
s.conn.RemotePeer(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
// Conn returns the Conn associated with this stream, as an network.Conn
|
|
|
|
func (s *Stream) Conn() network.Conn {
|
2018-07-04 10:51:47 +00:00
|
|
|
return s.conn
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read reads bytes from a stream.
|
|
|
|
func (s *Stream) Read(p []byte) (int, error) {
|
|
|
|
n, err := s.stream.Read(p)
|
|
|
|
// TODO: push this down to a lower level for better accuracy.
|
|
|
|
if s.conn.swarm.bwc != nil {
|
|
|
|
s.conn.swarm.bwc.LogRecvMessage(int64(n))
|
|
|
|
s.conn.swarm.bwc.LogRecvMessageStream(int64(n), s.Protocol(), s.Conn().RemotePeer())
|
|
|
|
}
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write writes bytes to a stream, flushing for each call.
|
|
|
|
func (s *Stream) Write(p []byte) (int, error) {
|
|
|
|
n, err := s.stream.Write(p)
|
|
|
|
// TODO: push this down to a lower level for better accuracy.
|
|
|
|
if s.conn.swarm.bwc != nil {
|
|
|
|
s.conn.swarm.bwc.LogSentMessage(int64(n))
|
|
|
|
s.conn.swarm.bwc.LogSentMessageStream(int64(n), s.Protocol(), s.Conn().RemotePeer())
|
|
|
|
}
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
// Close closes the stream, closing both ends and freeing all associated
|
|
|
|
// resources.
|
2018-07-04 10:51:47 +00:00
|
|
|
func (s *Stream) Close() error {
|
|
|
|
err := s.stream.Close()
|
2021-06-16 20:19:45 +00:00
|
|
|
s.closeOnce.Do(s.remove)
|
2018-07-04 10:51:47 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
// Reset resets the stream, signaling an error on both ends and freeing all
|
|
|
|
// associated resources.
|
2018-07-04 10:51:47 +00:00
|
|
|
func (s *Stream) Reset() error {
|
|
|
|
err := s.stream.Reset()
|
2021-06-16 20:19:45 +00:00
|
|
|
s.closeOnce.Do(s.remove)
|
2018-07-04 10:51:47 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
// Close closes the stream for writing, flushing all data and sending an EOF.
|
|
|
|
// This function does not free resources, call Close or Reset when done with the
|
|
|
|
// stream.
|
|
|
|
func (s *Stream) CloseWrite() error {
|
|
|
|
return s.stream.CloseWrite()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the stream for reading. This function does not free resources,
|
|
|
|
// call Close or Reset when done with the stream.
|
|
|
|
func (s *Stream) CloseRead() error {
|
|
|
|
return s.stream.CloseRead()
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
func (s *Stream) remove() {
|
|
|
|
s.conn.removeStream(s)
|
|
|
|
|
|
|
|
// We *must* do this in a goroutine. This can be called during a
|
|
|
|
// an open notification and will block until that notification is done.
|
|
|
|
go func() {
|
|
|
|
s.notifyLk.Lock()
|
|
|
|
defer s.notifyLk.Unlock()
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
s.conn.swarm.notifyAll(func(f network.Notifiee) {
|
2018-07-04 10:51:47 +00:00
|
|
|
f.ClosedStream(s.conn.swarm, s)
|
|
|
|
})
|
|
|
|
s.conn.swarm.refs.Done()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Protocol returns the protocol negotiated on this stream (if set).
|
|
|
|
func (s *Stream) Protocol() protocol.ID {
|
|
|
|
// Ignore type error. It means that the protocol is unset.
|
|
|
|
p, _ := s.protocol.Load().(protocol.ID)
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetProtocol sets the protocol for this stream.
|
|
|
|
//
|
|
|
|
// This doesn't actually *do* anything other than record the fact that we're
|
|
|
|
// speaking the given protocol over this stream. It's still up to the user to
|
|
|
|
// negotiate the protocol. This is usually done by the Host.
|
|
|
|
func (s *Stream) SetProtocol(p protocol.ID) {
|
|
|
|
s.protocol.Store(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetDeadline sets the read and write deadlines for this stream.
|
|
|
|
func (s *Stream) SetDeadline(t time.Time) error {
|
|
|
|
return s.stream.SetDeadline(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetReadDeadline sets the read deadline for this stream.
|
|
|
|
func (s *Stream) SetReadDeadline(t time.Time) error {
|
|
|
|
return s.stream.SetReadDeadline(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetWriteDeadline sets the write deadline for this stream.
|
|
|
|
func (s *Stream) SetWriteDeadline(t time.Time) error {
|
|
|
|
return s.stream.SetWriteDeadline(t)
|
|
|
|
}
|
2019-06-09 07:24:20 +00:00
|
|
|
|
|
|
|
// Stat returns metadata information for this stream.
|
|
|
|
func (s *Stream) Stat() network.Stat {
|
|
|
|
return s.stat
|
|
|
|
}
|