forked from TrueCloudLab/rclone
108 lines
1.9 KiB
Go
108 lines
1.9 KiB
Go
|
// This tests TimeoutReader
|
||
|
|
||
|
package swift
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
"io/ioutil"
|
||
|
"sync"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// An io.ReadCloser for testing
|
||
|
type testReader struct {
|
||
|
sync.Mutex
|
||
|
n int
|
||
|
delay time.Duration
|
||
|
closed bool
|
||
|
}
|
||
|
|
||
|
// Returns n bytes with at time.Duration delay
|
||
|
func newTestReader(n int, delay time.Duration) *testReader {
|
||
|
return &testReader{
|
||
|
n: n,
|
||
|
delay: delay,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Returns 1 byte at a time after delay
|
||
|
func (t *testReader) Read(p []byte) (n int, err error) {
|
||
|
if t.n <= 0 {
|
||
|
return 0, io.EOF
|
||
|
}
|
||
|
time.Sleep(t.delay)
|
||
|
p[0] = 'A'
|
||
|
t.Lock()
|
||
|
t.n--
|
||
|
t.Unlock()
|
||
|
return 1, nil
|
||
|
}
|
||
|
|
||
|
// Close the channel
|
||
|
func (t *testReader) Close() error {
|
||
|
t.Lock()
|
||
|
t.closed = true
|
||
|
t.Unlock()
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func TestTimeoutReaderNoTimeout(t *testing.T) {
|
||
|
test := newTestReader(3, 10*time.Millisecond)
|
||
|
cancelled := false
|
||
|
cancel := func() {
|
||
|
cancelled = true
|
||
|
}
|
||
|
tr := newTimeoutReader(test, 100*time.Millisecond, cancel)
|
||
|
b, err := ioutil.ReadAll(tr)
|
||
|
if err != nil || string(b) != "AAA" {
|
||
|
t.Fatalf("Bad read %s %s", err, b)
|
||
|
}
|
||
|
if cancelled {
|
||
|
t.Fatal("Cancelled when shouldn't have been")
|
||
|
}
|
||
|
if test.n != 0 {
|
||
|
t.Fatal("Didn't read all")
|
||
|
}
|
||
|
if test.closed {
|
||
|
t.Fatal("Shouldn't be closed")
|
||
|
}
|
||
|
tr.Close()
|
||
|
if !test.closed {
|
||
|
t.Fatal("Should be closed")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestTimeoutReaderTimeout(t *testing.T) {
|
||
|
// Return those bytes slowly so we get an idle timeout
|
||
|
test := newTestReader(3, 100*time.Millisecond)
|
||
|
cancelled := false
|
||
|
cancel := func() {
|
||
|
cancelled = true
|
||
|
}
|
||
|
tr := newTimeoutReader(test, 10*time.Millisecond, cancel)
|
||
|
_, err := ioutil.ReadAll(tr)
|
||
|
if err != TimeoutError {
|
||
|
t.Fatal("Expecting TimeoutError, got", err)
|
||
|
}
|
||
|
if !cancelled {
|
||
|
t.Fatal("Not cancelled when should have been")
|
||
|
}
|
||
|
test.Lock()
|
||
|
n := test.n
|
||
|
test.Unlock()
|
||
|
if n == 0 {
|
||
|
t.Fatal("Read all")
|
||
|
}
|
||
|
if n != 3 {
|
||
|
t.Fatal("Didn't read any")
|
||
|
}
|
||
|
if test.closed {
|
||
|
t.Fatal("Shouldn't be closed")
|
||
|
}
|
||
|
tr.Close()
|
||
|
if !test.closed {
|
||
|
t.Fatal("Should be closed")
|
||
|
}
|
||
|
}
|