package quic import ( "net" "time" "github.com/lucas-clemente/quic-go/internal/flowcontrol" "github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/wire" ) const ( errorCodeStopping protocol.ApplicationErrorCode = 0 errorCodeStoppingGQUIC protocol.ApplicationErrorCode = 7 ) type streamI interface { Stream HandleStreamFrame(*wire.StreamFrame) error HandleRstStreamFrame(*wire.RstStreamFrame) error HandleStopSendingFrame(*wire.StopSendingFrame) PopStreamFrame(maxBytes protocol.ByteCount) *wire.StreamFrame Finished() bool CloseForShutdown(error) // methods needed for flow control GetWindowUpdate() protocol.ByteCount HandleMaxStreamDataFrame(*wire.MaxStreamDataFrame) } // A Stream assembles the data from StreamFrames and provides a super-convenient Read-Interface // // Read() and Write() may be called concurrently, but multiple calls to Read() or Write() individually must be synchronized manually. type stream struct { receiveStream sendStream version protocol.VersionNumber } var _ Stream = &stream{} var _ streamI = &stream{} type deadlineError struct{} func (deadlineError) Error() string { return "deadline exceeded" } func (deadlineError) Temporary() bool { return true } func (deadlineError) Timeout() bool { return true } var errDeadline net.Error = &deadlineError{} type streamCanceledError struct { error errorCode protocol.ApplicationErrorCode } func (streamCanceledError) Canceled() bool { return true } func (e streamCanceledError) ErrorCode() protocol.ApplicationErrorCode { return e.errorCode } var _ StreamError = &streamCanceledError{} // newStream creates a new Stream func newStream(streamID protocol.StreamID, onData func(), queueControlFrame func(wire.Frame), flowController flowcontrol.StreamFlowController, version protocol.VersionNumber, ) *stream { return &stream{ sendStream: *newSendStream(streamID, onData, queueControlFrame, flowController, version), receiveStream: *newReceiveStream(streamID, onData, queueControlFrame, flowController), } } // need to define StreamID() here, since both receiveStream and readStream have a StreamID() func (s *stream) StreamID() protocol.StreamID { // the result is same for receiveStream and sendStream return s.sendStream.StreamID() } func (s *stream) Close() error { if err := s.sendStream.Close(); err != nil { return err } // in gQUIC, we need to send a RST_STREAM with the final offset if CancelRead() was called s.receiveStream.onClose(s.sendStream.getWriteOffset()) return nil } func (s *stream) SetDeadline(t time.Time) error { _ = s.SetReadDeadline(t) // SetReadDeadline never errors _ = s.SetWriteDeadline(t) // SetWriteDeadline never errors return nil } // CloseForShutdown closes a stream abruptly. // It makes Read and Write unblock (and return the error) immediately. // The peer will NOT be informed about this: the stream is closed without sending a FIN or RST. func (s *stream) CloseForShutdown(err error) { s.sendStream.CloseForShutdown(err) s.receiveStream.CloseForShutdown(err) } func (s *stream) HandleRstStreamFrame(frame *wire.RstStreamFrame) error { if err := s.receiveStream.HandleRstStreamFrame(frame); err != nil { return err } if !s.version.UsesIETFFrameFormat() { s.HandleStopSendingFrame(&wire.StopSendingFrame{ StreamID: s.StreamID(), ErrorCode: frame.ErrorCode, }) } return nil } func (s *stream) Finished() bool { return s.sendStream.Finished() && s.receiveStream.Finished() }