mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
124 lines
3.5 KiB
Go
124 lines
3.5 KiB
Go
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
|
|
)
|
|
|
|
// The streamSender is notified by the stream about various events.
|
|
type streamSender interface {
|
|
scheduleSending()
|
|
queueControlFrame(wire.Frame)
|
|
}
|
|
|
|
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,
|
|
sender streamSender,
|
|
flowController flowcontrol.StreamFlowController,
|
|
version protocol.VersionNumber,
|
|
) *stream {
|
|
return &stream{
|
|
sendStream: *newSendStream(streamID, sender, flowController, version),
|
|
receiveStream: *newReceiveStream(streamID, sender, 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()
|
|
}
|