rclone/vendor/storj.io/drpc/drpcstream/stream.go

425 lines
11 KiB
Go
Raw Normal View History

2020-05-11 18:57:46 +00:00
// Copyright (C) 2019 Storj Labs, Inc.
// See LICENSE for copying information.
package drpcstream
import (
"context"
"fmt"
"io"
"sync"
"github.com/gogo/protobuf/proto"
"github.com/zeebo/errs"
"storj.io/drpc"
"storj.io/drpc/drpcdebug"
"storj.io/drpc/drpcsignal"
"storj.io/drpc/drpcwire"
)
// Options controls configuration settings for a stream.
type Options struct {
// SplitSize controls the default size we split packets into frames.
SplitSize int
}
// Stream represents an rpc actively happening on a transport.
type Stream struct {
ctx context.Context
cancel func()
opts Options
writeMu chMutex
id drpcwire.ID
wr *drpcwire.Writer
mu sync.Mutex // protects state transitions
sigs struct {
send drpcsignal.Signal // set when done sending messages
recv drpcsignal.Signal // set when done receiving messages
term drpcsignal.Signal // set when in terminated state
finish drpcsignal.Signal // set when all writes are complete
cancel drpcsignal.Signal // set when externally canceled and transport will be closed
}
queue chan drpcwire.Packet
// avoids allocations of closures
pollWriteFn func(drpcwire.Frame) error
}
var _ drpc.Stream = (*Stream)(nil)
// New returns a new stream bound to the context with the given stream id and will
// use the writer to write messages on. It is important use monotonically increasing
// stream ids within a single transport.
func New(ctx context.Context, sid uint64, wr *drpcwire.Writer) *Stream {
return NewWithOptions(ctx, sid, wr, Options{})
}
// NewWithOptions returns a new stream bound to the context with the given stream id
// and will use the writer to write messages on. It is important use monotonically increasing
// stream ids within a single transport. The options are used to control details of how
// the Stream operates.
func NewWithOptions(ctx context.Context, sid uint64, wr *drpcwire.Writer, opts Options) *Stream {
ctx, cancel := context.WithCancel(ctx)
s := &Stream{
ctx: ctx,
cancel: cancel,
opts: opts,
wr: wr,
id: drpcwire.ID{Stream: sid},
queue: make(chan drpcwire.Packet),
}
s.pollWriteFn = s.pollWrite
return s
}
//
// monitoring helpers
//
// monCtx returns a copy of the context for use with mon.Task so that there aren't
// races overwriting the stream's context.
func (s *Stream) monCtx() *context.Context {
ctx := s.ctx
return &ctx
}
//
// accessors
//
// Context returns the context associated with the stream. It is closed when
// the Stream will no longer issue any writes or reads.
func (s *Stream) Context() context.Context { return s.ctx }
// Terminated returns a channel when the stream has been terminated.
func (s *Stream) Terminated() <-chan struct{} { return s.sigs.term.Signal() }
// Finished returns true if the stream is fully finished and will no longer
// issue any writes or reads.
func (s *Stream) Finished() bool { return s.sigs.finish.IsSet() }
//
// packet handler
//
// HandlePacket advances the stream state machine by inspecting the packet. It returns
// any major errors that should terminate the transport the stream is operating on as
// well as a boolean indicating if the stream expects more packets.
func (s *Stream) HandlePacket(pkt drpcwire.Packet) (more bool, err error) {
defer mon.Task()(s.monCtx())(&err)
s.mu.Lock()
defer s.mu.Unlock()
drpcdebug.Log(func() string { return fmt.Sprintf("STR[%p][%d]: %v", s, s.id.Stream, pkt) })
if pkt.ID.Stream != s.id.Stream {
return true, nil
}
switch pkt.Kind {
case drpcwire.KindInvoke:
err := drpc.ProtocolError.New("invoke on existing stream")
s.terminate(err)
return false, err
case drpcwire.KindMessage:
if s.sigs.recv.IsSet() || s.sigs.term.IsSet() {
return true, nil
}
// drop the mutex while we either send into the queue or we're told that
// receiving is done. we don't handle any more packets until the message
// is delivered, so the only way it can become set is from some of the
// stream terminating calls, in which case, shutting down the stream is
// racing with the message being received, so dropping it is valid.
s.mu.Unlock()
defer s.mu.Lock()
select {
case <-s.sigs.recv.Signal():
case <-s.sigs.term.Signal():
case s.queue <- pkt:
}
return true, nil
case drpcwire.KindError:
err := drpcwire.UnmarshalError(pkt.Data)
s.sigs.send.Set(io.EOF) // in this state, gRPC returns io.EOF on send.
s.terminate(err)
return false, nil
case drpcwire.KindClose:
s.sigs.recv.Set(io.EOF)
s.terminate(drpc.Error.New("remote closed the stream"))
return false, nil
case drpcwire.KindCloseSend:
s.sigs.recv.Set(io.EOF)
s.terminateIfBothClosed()
return false, nil
default:
err := drpc.InternalError.New("unknown packet kind: %s", pkt.Kind)
s.terminate(err)
return false, err
}
}
//
// helpers
//
// checkFinished checks to see if the stream is terminated, and if so, sets the finished
// flag. This must be called every time right before we release the write mutex.
func (s *Stream) checkFinished() {
if s.sigs.term.IsSet() {
s.sigs.finish.Set(nil)
}
}
// checkCancelError will replace the error with one from the cancel signal if it is
// set. This is to prevent errors from reads/writes to a transport after it has been
// asynchronously closed due to context cancelation.
func (s *Stream) checkCancelError(err error) error {
if sigErr, ok := s.sigs.cancel.Get(); ok {
return sigErr
}
return err
}
// newPackage bumps the internal message id and returns a packet. It must be called
// under a mutex.
func (s *Stream) newPacket(kind drpcwire.Kind, data []byte) drpcwire.Packet {
s.id.Message++
return drpcwire.Packet{
Data: data,
ID: s.id,
Kind: kind,
}
}
// pollWrite checks for any conditions that should cause a write to not happen and
// then issues the write of the frame.
func (s *Stream) pollWrite(fr drpcwire.Frame) (err error) {
switch {
case s.sigs.send.IsSet():
return s.sigs.send.Err()
case s.sigs.term.IsSet():
return s.sigs.term.Err()
}
return s.checkCancelError(errs.Wrap(s.wr.WriteFrame(fr)))
}
// sendPacket sends the packet in a single write and flushes. It does not check for
// any conditions to stop it from writing and is meant for internal stream use to
// do things like signal errors or closes to the remote side.
func (s *Stream) sendPacket(kind drpcwire.Kind, data []byte) (err error) {
defer mon.Task()(s.monCtx())(&err)
if err := s.wr.WritePacket(s.newPacket(kind, data)); err != nil {
return errs.Wrap(err)
}
if err := s.wr.Flush(); err != nil {
return errs.Wrap(err)
}
return nil
}
// terminateIfBothClosed is a helper to terminate the stream if both sides have
// issued a CloseSend.
func (s *Stream) terminateIfBothClosed() {
if s.sigs.send.IsSet() && s.sigs.recv.IsSet() {
s.terminate(drpc.Error.New("stream terminated by both issuing close send"))
}
}
// terminate marks the stream as terminated with the given error. It also marks
// the stream as finished if no writes are happening at the time of the call.
func (s *Stream) terminate(err error) {
s.sigs.send.Set(err)
s.sigs.recv.Set(err)
s.sigs.term.Set(err)
s.cancel()
// if we can acquire the write mutex, then checkFinished. if not, then we know
// some other write is happening, and it will call checkFinished before it
// releases the mutex.
if s.writeMu.TryLock() {
s.checkFinished()
s.writeMu.Unlock()
}
}
//
// raw read/write
//
// RawWrite sends the data bytes with the given kind.
func (s *Stream) RawWrite(kind drpcwire.Kind, data []byte) (err error) {
defer mon.Task()(s.monCtx())(&err)
s.writeMu.Lock()
defer s.writeMu.Unlock()
defer s.checkFinished()
return drpcwire.SplitN(s.newPacket(kind, data), s.opts.SplitSize, s.pollWriteFn)
}
// RawFlush flushes any buffers of data.
func (s *Stream) RawFlush() (err error) {
defer mon.Task()(s.monCtx())(&err)
s.writeMu.Lock()
defer s.writeMu.Unlock()
defer s.checkFinished()
return s.checkCancelError(errs.Wrap(s.wr.Flush()))
}
// RawRecv returns the raw bytes received for a message.
func (s *Stream) RawRecv() (data []byte, err error) {
defer mon.Task()(s.monCtx())(&err)
if s.sigs.recv.IsSet() {
return nil, s.sigs.recv.Err()
}
select {
case <-s.sigs.recv.Signal():
return nil, s.sigs.recv.Err()
case pkt := <-s.queue:
return pkt.Data, nil
}
}
//
// msg read/write
//
// MsgSend marshals the message with protobuf, writes it, and flushes.
func (s *Stream) MsgSend(msg drpc.Message) (err error) {
defer mon.Task()(s.monCtx())(&err)
data, err := proto.Marshal(msg)
if err != nil {
return errs.Wrap(err)
}
if err := s.RawWrite(drpcwire.KindMessage, data); err != nil {
return err
}
if err := s.RawFlush(); err != nil {
return err
}
return nil
}
// MsgRecv recives some protobuf data and unmarshals it into msg.
func (s *Stream) MsgRecv(msg drpc.Message) (err error) {
defer mon.Task()(s.monCtx())(&err)
data, err := s.RawRecv()
if err != nil {
return err
}
return proto.Unmarshal(data, msg)
}
//
// terminal messages
//
// SendError terminates the stream and sends the error to the remote. It is a no-op if
// the stream is already terminated.
func (s *Stream) SendError(serr error) (err error) {
defer mon.Task()(s.monCtx())(&err)
s.mu.Lock()
if s.sigs.term.IsSet() {
s.mu.Unlock()
return nil
}
s.writeMu.Lock()
defer s.writeMu.Unlock()
defer s.checkFinished()
s.sigs.send.Set(io.EOF) // in this state, gRPC returns io.EOF on send.
s.terminate(drpc.Error.New("stream terminated by sending error"))
s.mu.Unlock()
return s.checkCancelError(s.sendPacket(drpcwire.KindError, drpcwire.MarshalError(serr)))
}
// Close terminates the stream and sends that the stream has been closed to the remote.
// It is a no-op if the stream is already terminated.
func (s *Stream) Close() (err error) {
defer mon.Task()(s.monCtx())(&err)
s.mu.Lock()
if s.sigs.term.IsSet() {
s.mu.Unlock()
return nil
}
s.writeMu.Lock()
defer s.writeMu.Unlock()
defer s.checkFinished()
s.terminate(drpc.Error.New("stream terminated by sending close"))
s.mu.Unlock()
return s.checkCancelError(s.sendPacket(drpcwire.KindClose, nil))
}
// CloseSend informs the remote that no more messages will be sent. If the remote has
// also already issued a CloseSend, the stream is terminated. It is a no-op if the
// stream already has sent a CloseSend or if it is terminated.
func (s *Stream) CloseSend() (err error) {
defer mon.Task()(s.monCtx())(&err)
s.mu.Lock()
if s.sigs.send.IsSet() || s.sigs.term.IsSet() {
s.mu.Unlock()
return nil
}
s.writeMu.Lock()
defer s.writeMu.Unlock()
defer s.checkFinished()
s.sigs.send.Set(drpc.Error.New("send closed"))
s.terminateIfBothClosed()
s.mu.Unlock()
return s.checkCancelError(s.sendPacket(drpcwire.KindCloseSend, nil))
}
// Cancel transitions the stream into a state where all writes to the transport will return
// the provided error, and terminates the stream. It is a no-op if the stream is already
// terminated.
func (s *Stream) Cancel(err error) {
defer mon.Task()(s.monCtx())(nil)
s.mu.Lock()
defer s.mu.Unlock()
if s.sigs.term.IsSet() {
return
}
s.sigs.cancel.Set(err)
s.sigs.send.Set(io.EOF) // in this state, gRPC returns io.EOF on send.
s.terminate(err)
}