zrepl/util/contextflexibletimeout_test.go

85 lines
2.4 KiB
Go
Raw Normal View History

2018-07-15 17:36:53 +02:00
package util
import (
"context"
"github.com/stretchr/testify/assert"
2018-08-25 21:30:25 +02:00
"github.com/stretchr/testify/require"
"testing"
"time"
2018-07-15 17:36:53 +02:00
)
func TestContextWithOptionalDeadline(t *testing.T) {
ctx := context.Background()
cctx, enforceDeadline := ContextWithOptionalDeadline(ctx)
begin := time.Now()
var receivedCancellation time.Time
var cancellationError error
go func() {
select {
2018-08-25 21:30:25 +02:00
case <-cctx.Done():
2018-07-15 17:36:53 +02:00
receivedCancellation = time.Now()
cancellationError = cctx.Err()
2018-08-25 21:30:25 +02:00
case <-time.After(600 * time.Millisecond):
2018-07-15 17:36:53 +02:00
t.Fatalf("should have been cancelled by deadline")
}
}()
2018-08-25 21:30:25 +02:00
time.Sleep(100 * time.Millisecond)
2018-07-15 17:36:53 +02:00
if !receivedCancellation.IsZero() {
t.Fatalf("no enforcement means no cancellation")
}
require.Nil(t, cctx.Err(), "no error while not cancelled")
dl, ok := cctx.Deadline()
require.False(t, ok)
require.Zero(t, dl)
2018-08-25 21:30:25 +02:00
enforceDeadline(begin.Add(200 * time.Millisecond))
2018-07-15 17:36:53 +02:00
// second call must be ignored, i.e. we expect the deadline to be at begin+200ms, not begin+400ms
2018-08-25 21:30:25 +02:00
enforceDeadline(begin.Add(400 * time.Millisecond))
2018-07-15 17:36:53 +02:00
2018-08-25 21:30:25 +02:00
time.Sleep(300 * time.Millisecond) // 100ms margin for scheduler
2018-07-15 17:36:53 +02:00
if receivedCancellation.Sub(begin) > 250*time.Millisecond {
t.Fatalf("cancellation is beyond acceptable scheduler latency")
}
require.Equal(t, context.DeadlineExceeded, cancellationError)
}
func TestContextWithOptionalDeadlineNegativeDeadline(t *testing.T) {
ctx := context.Background()
cctx, enforceDeadline := ContextWithOptionalDeadline(ctx)
2018-08-25 21:30:25 +02:00
enforceDeadline(time.Now().Add(-10 * time.Second))
2018-07-15 17:36:53 +02:00
select {
case <-cctx.Done():
default:
t.FailNow()
}
}
func TestContextWithOptionalDeadlineParentCancellation(t *testing.T) {
pctx, cancel := context.WithCancel(context.Background())
cctx, enforceDeadline := ContextWithOptionalDeadline(pctx)
// 0 ms
start := time.Now()
2018-08-25 21:30:25 +02:00
enforceDeadline(start.Add(400 * time.Millisecond))
time.Sleep(100 * time.Millisecond)
cancel() // cancel @ ~100ms
time.Sleep(100 * time.Millisecond) // give 100ms time to propagate cancel
2018-07-15 17:36:53 +02:00
// @ ~200ms
select {
case <-cctx.Done():
assert.True(t, time.Now().Before(start.Add(300*time.Millisecond)))
assert.Equal(t, context.Canceled, cctx.Err())
default:
t.FailNow()
}
}
func TestContextWithOptionalDeadlineValue(t *testing.T) {
pctx := context.WithValue(context.Background(), "key", "value")
cctx, _ := ContextWithOptionalDeadline(pctx)
assert.Equal(t, "value", cctx.Value("key"))
2018-08-25 21:30:25 +02:00
}