Skip to content

Commit

Permalink
v1,v2: Return available data alongside error
Browse files Browse the repository at this point in the history
  • Loading branch information
lukechampine committed Jun 1, 2023
1 parent 537577b commit 12392de
Show file tree
Hide file tree
Showing 4 changed files with 52 additions and 31 deletions.
37 changes: 24 additions & 13 deletions v1/mux.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import (
"crypto/ed25519"
"errors"
"fmt"
"io"
"net"
"os"
"sync"
Expand Down Expand Up @@ -59,6 +60,7 @@ func (m *Mux) setErr(err error) error {
for _, s := range m.streams {
s.cond.L.Lock()
s.err = err
s.readBuf = nil
s.cond.Broadcast()
s.cond.L.Unlock()
}
Expand Down Expand Up @@ -381,34 +383,42 @@ func (s *Stream) consumeFrame(h frameHeader, payload []byte) {
// set payload and wait for it to be consumed
s.readBuf = payload
s.cond.Broadcast() // wake Read
for len(s.readBuf) != 0 && s.err == nil {
for len(s.readBuf) > 0 && s.err == nil {
s.cond.Wait()
}
if len(s.readBuf) != 0 {
// should never happen
panic("consumeFrame must not return before frame has been fully consumed")
}
}

// Read reads data from the Stream.
func (s *Stream) Read(p []byte) (int, error) {
s.cond.L.Lock()
defer s.cond.L.Unlock()
if !s.rd.IsZero() {
if !time.Now().Before(s.rd) {
return 0, os.ErrDeadlineExceeded
}
timer := time.AfterFunc(time.Until(s.rd), s.cond.Broadcast)
defer timer.Stop()
defer time.AfterFunc(time.Until(s.rd), s.cond.Broadcast).Stop()
}
for len(s.readBuf) == 0 && s.err == nil && (s.rd.IsZero() || time.Now().Before(s.rd)) {
s.cond.Wait()
}
if s.err != nil {
return 0, s.err
} else if !s.rd.IsZero() && !time.Now().Before(s.rd) {
return 0, os.ErrDeadlineExceeded
}
n := copy(p, s.readBuf)
s.readBuf = s.readBuf[n:]
s.cond.Broadcast() // wake consumeFrame
return n, s.err

err := s.err
if err == ErrPeerClosedStream {
err = io.EOF
} else if !(s.rd.IsZero() || time.Now().Before(s.rd)) {
err = os.ErrDeadlineExceeded
} else if err != nil {
s.readBuf = nil // if the error is fatal, drop the rest of the buffer
}
if len(s.readBuf) > 0 {
err = nil // if more data is available, silence the error
} else {
s.cond.Broadcast() // wake consumeFrame
}
return n, err
}

// Write writes data to the Stream.
Expand Down Expand Up @@ -452,6 +462,7 @@ func (s *Stream) Close() error {
s.cond.L.Lock()
defer s.cond.L.Unlock()
s.err = ErrClosedStream
s.readBuf = nil
s.cond.Broadcast()
return err
}
Expand Down
2 changes: 1 addition & 1 deletion v1/mux_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -262,7 +262,7 @@ func TestDeadline(t *testing.T) {

// need to write a fairly large message; otherwise the packets just
// get buffered and "succeed" instantly
if _, err := s.Write(make([]byte, 1<<20)); err != nil {
if _, err := s.Write(make([]byte, m.settings.RequestedPacketSize*20)); err != nil {
return err
} else if _, err := io.ReadFull(s, buf[:13]); err != nil {
return err
Expand Down
42 changes: 26 additions & 16 deletions v2/mux.go
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ func (m *Mux) setErr(err error) error {
for _, s := range m.streams {
s.cond.L.Lock()
s.err = err
s.readBuf = nil
s.cond.Broadcast()
s.cond.L.Unlock()
}
Expand Down Expand Up @@ -490,11 +491,18 @@ func (s *Stream) consumeFrame(h frameHeader, payload []byte) {
// set payload and wait for it to be consumed
s.cond.L.Lock()
defer s.cond.L.Unlock()
if s.err != nil {
return
}
s.readBuf = payload
s.cond.Broadcast() // wake Read
for len(s.readBuf) > 0 && s.err == nil && (s.rd.IsZero() || time.Now().Before(s.rd)) {
for len(s.readBuf) > 0 && s.err == nil {
s.cond.Wait()
}
if len(s.readBuf) != 0 {
// should never happen
panic("consumeFrame must not return before frame has been fully consumed")
}
}

// Read reads data from the Stream.
Expand All @@ -506,27 +514,28 @@ func (s *Stream) Read(p []byte) (int, error) {
panic("mux: Read called before Write on newly-Dialed Stream")
}
if !s.rd.IsZero() {
if !time.Now().Before(s.rd) {
return 0, os.ErrDeadlineExceeded
}
timer := time.AfterFunc(time.Until(s.rd), s.cond.Broadcast)
defer timer.Stop()
defer time.AfterFunc(time.Until(s.rd), s.cond.Broadcast).Stop()
}
for len(s.readBuf) == 0 && s.err == nil && (s.rd.IsZero() || time.Now().Before(s.rd)) {
s.cond.Wait()
}
if s.err != nil {
if s.err == ErrPeerClosedStream {
return 0, io.EOF
}
return 0, s.err
} else if !s.rd.IsZero() && !time.Now().Before(s.rd) {
return 0, os.ErrDeadlineExceeded
}
n := copy(p, s.readBuf)
s.readBuf = s.readBuf[n:]
s.cond.Broadcast() // wake consumeFrame
return n, nil

err := s.err
if err == ErrPeerClosedStream {
err = io.EOF
} else if !(s.rd.IsZero() || time.Now().Before(s.rd)) {
err = os.ErrDeadlineExceeded
} else if err != nil {
s.readBuf = nil // if the error is fatal, drop the rest of the buffer
}
if len(s.readBuf) > 0 {
err = nil // if more data is available, silence the error
} else {
s.cond.Broadcast() // wake consumeFrame
}
return n, err
}

// Write writes data to the Stream.
Expand Down Expand Up @@ -573,6 +582,7 @@ func (s *Stream) Close() error {
return nil
}
s.err = ErrClosedStream
s.readBuf = nil
s.cond.Broadcast()
s.cond.L.Unlock()

Expand Down
2 changes: 1 addition & 1 deletion v2/mux_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -277,7 +277,7 @@ func TestDeadline(t *testing.T) {
// need to write a fairly large message; otherwise the packets just
// get buffered and "succeed" instantly
if _, err := s.Write(make([]byte, m1.settings.PacketSize*20)); err != nil {
return fmt.Errorf("foo: %w", err)
return err
} else if _, err := io.ReadFull(s, buf[:13]); err != nil {
return err
} else if string(buf) != "hello, world!" {
Expand Down

0 comments on commit 12392de

Please sign in to comment.