blob: cdc7e0e798d35738c0530a7eea4b978f56a7670a [file] [log] [blame]
Matt Rosencrantz137b8d22014-08-18 09:56:15 -07001package ipc
2
3import (
4 "sync"
5 "testing"
6 "time"
7
Jiri Simsa519c5072014-09-17 21:37:57 -07008 "veyron.io/veyron/veyron/runtimes/google/testing/mocks/runtime"
9 "veyron.io/veyron/veyron/runtimes/google/vtrace"
Matt Rosencrantz9fe60822014-09-12 10:09:53 -070010
Jiri Simsa519c5072014-09-17 21:37:57 -070011 "veyron.io/veyron/veyron2/context"
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070012)
13
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070014// We need a special way to create contexts for tests. We
15// can't create a real runtime in the runtime implementation
16// so we use a fake one that panics if used. The runtime
17// implementation should not ever use the Runtime from a context.
18func testContext() context.T {
Matt Rosencrantzcc922c12014-11-28 20:28:59 -080019 ctx, _ := testContextWithoutDeadline().WithTimeout(20 * time.Second)
20 return ctx
21}
22
23func testContextWithoutDeadline() context.T {
Matt Rosencrantz9fe60822014-09-12 10:09:53 -070024 ctx := InternalNewContext(&runtime.PanicRuntime{})
Matt Rosencrantz3e76f282014-11-10 09:38:57 -080025 ctx, _ = vtrace.WithNewRootSpan(ctx, nil, false)
Matt Rosencrantz9fe60822014-09-12 10:09:53 -070026 return ctx
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070027}
28
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070029func testCancel(t *testing.T, ctx context.T, cancel context.CancelFunc) {
30 select {
31 case <-ctx.Done():
32 t.Errorf("Done closed when deadline not yet passed")
33 default:
34 }
35 ch := make(chan bool, 0)
36 go func() {
37 cancel()
38 close(ch)
39 }()
40 select {
41 case <-ch:
42 case <-time.After(3 * time.Second):
43 t.Fatal("timed out witing for cancel.")
44 }
45
46 select {
47 case <-ctx.Done():
48 case <-time.After(3 * time.Second):
49 t.Fatal("timed out witing for cancellation.")
50 }
51 if err := ctx.Err(); err != context.Canceled {
52 t.Errorf("Unexpected error want %v, got %v", context.Canceled, err)
53 }
54}
55
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -080056func TestRootContext(t *testing.T) {
57 r := &runtime.PanicRuntime{}
58 ctx := InternalNewContext(r)
59
60 if got := ctx.Runtime(); got != r {
61 t.Errorf("Expected runtime %v, but found %v", r, got)
62 }
63
64 if got := ctx.Err(); got != nil {
65 t.Errorf("Expected nil error, got: %v", got)
66 }
67
68 defer func() {
69 r := recover()
70 if r != nilRuntimeMessage {
71 t.Errorf("Unexpected recover value: %s", r)
72 }
73 }()
74 InternalNewContext(nil)
75}
76
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070077func TestCancelContext(t *testing.T) {
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070078 ctx, cancel := testContext().WithCancel()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070079 testCancel(t, ctx, cancel)
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -080080
81 // Test cancelling a cancel context which is the child
82 // of a cancellable context.
83 parent, _ := testContext().WithCancel()
84 child, cancel := parent.WithCancel()
85 cancel()
86 <-child.Done()
87
88 // Test adding a cancellable child context after the parent is
89 // already cancelled.
90 parent, cancel = testContext().WithCancel()
91 cancel()
92 child, _ = parent.WithCancel()
93 <-child.Done() // The child should have been cancelled right away.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070094}
95
96func TestMultiLevelCancelContext(t *testing.T) {
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070097 c0, c0Cancel := testContext().WithCancel()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070098 c1, _ := c0.WithCancel()
99 c2, _ := c1.WithCancel()
100 c3, _ := c2.WithCancel()
101 testCancel(t, c3, c0Cancel)
102}
103
104type nonStandardContext struct {
105 context.T
106}
107
108func (n *nonStandardContext) WithCancel() (ctx context.T, cancel context.CancelFunc) {
109 return newCancelContext(n)
110}
111func (n *nonStandardContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) {
112 return newDeadlineContext(n, deadline)
113}
114func (n *nonStandardContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) {
115 return newDeadlineContext(n, time.Now().Add(timeout))
116}
117func (n *nonStandardContext) WithValue(key interface{}, val interface{}) context.T {
118 return newValueContext(n, key, val)
119}
120
121func TestCancelContextWithNonStandard(t *testing.T) {
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800122 // Test that cancellation flows properly through non-standard intermediates.
123 ctx := testContext()
124 c0 := &nonStandardContext{ctx}
125 c1, c1Cancel := c0.WithCancel()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700126 c2 := &nonStandardContext{c1}
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800127 c3 := &nonStandardContext{c2}
128 c4, _ := c3.WithCancel()
129 testCancel(t, c4, c1Cancel)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700130}
131
132func testDeadline(t *testing.T, ctx context.T, start time.Time, desiredTimeout time.Duration) {
133 <-ctx.Done()
134 if delta := time.Now().Sub(start); delta < desiredTimeout {
135 t.Errorf("Deadline too short want %s got %s", desiredTimeout, delta)
136 }
137 if err := ctx.Err(); err != context.DeadlineExceeded {
138 t.Errorf("Unexpected error want %s, got %s", context.DeadlineExceeded, err)
139 }
140}
141
142func TestDeadlineContext(t *testing.T) {
143 cases := []time.Duration{
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800144 3 * time.Millisecond,
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700145 0,
146 }
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800147 rootCtx := InternalNewContext(&runtime.PanicRuntime{})
148 cancelCtx, _ := rootCtx.WithCancel()
149 deadlineCtx, _ := rootCtx.WithDeadline(time.Now().Add(time.Hour))
150
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700151 for _, desiredTimeout := range cases {
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800152 // Test all the various ways of getting deadline contexts.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700153 start := time.Now()
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800154 ctx, _ := rootCtx.WithDeadline(start.Add(desiredTimeout))
155 testDeadline(t, ctx, start, desiredTimeout)
156
157 start = time.Now()
158 ctx, _ = cancelCtx.WithDeadline(start.Add(desiredTimeout))
159 testDeadline(t, ctx, start, desiredTimeout)
160
161 start = time.Now()
162 ctx, _ = deadlineCtx.WithDeadline(start.Add(desiredTimeout))
163 testDeadline(t, ctx, start, desiredTimeout)
164
165 start = time.Now()
166 ctx, _ = rootCtx.WithTimeout(desiredTimeout)
167 testDeadline(t, ctx, start, desiredTimeout)
168
169 start = time.Now()
170 ctx, _ = cancelCtx.WithTimeout(desiredTimeout)
171 testDeadline(t, ctx, start, desiredTimeout)
172
173 start = time.Now()
174 ctx, _ = deadlineCtx.WithTimeout(desiredTimeout)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700175 testDeadline(t, ctx, start, desiredTimeout)
176 }
177
Matt Rosencrantzbf85d542014-08-22 13:31:14 -0700178 ctx, cancel := testContext().WithDeadline(time.Now().Add(100 * time.Hour))
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700179 testCancel(t, ctx, cancel)
180}
181
182func TestDeadlineContextWithRace(t *testing.T) {
Matt Rosencrantzbf85d542014-08-22 13:31:14 -0700183 ctx, cancel := testContext().WithDeadline(time.Now().Add(100 * time.Hour))
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700184 var wg sync.WaitGroup
185 wg.Add(10)
186 for i := 0; i < 10; i++ {
187 go func() {
188 cancel()
189 wg.Done()
190 }()
191 }
192 wg.Wait()
193 <-ctx.Done()
194 if err := ctx.Err(); err != context.Canceled {
195 t.Errorf("Unexpected error want %v, got %v", context.Canceled, err)
196 }
197}
198
199func TestValueContext(t *testing.T) {
200 type testContextKey int
201 const (
202 key1 = testContextKey(iota)
203 key2
204 key3
205 key4
206 )
207 const (
208 val1 = iota
209 val2
210 val3
211 )
Matt Rosencrantzbf85d542014-08-22 13:31:14 -0700212 ctx1 := testContext().WithValue(key1, val1)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700213 ctx2 := ctx1.WithValue(key2, val2)
214 ctx3 := ctx2.WithValue(key3, val3)
215
216 expected := map[interface{}]interface{}{
217 key1: val1,
218 key2: val2,
219 key3: val3,
220 key4: nil,
221 }
222 for k, v := range expected {
223 if got := ctx3.Value(k); got != v {
224 t.Errorf("Got wrong value for %v: want %v got %v", k, v, got)
225 }
226 }
227
228}