Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 1 | package ipc |
| 2 | |
| 3 | import ( |
| 4 | "sync" |
| 5 | "testing" |
| 6 | "time" |
| 7 | |
Jiri Simsa | 519c507 | 2014-09-17 21:37:57 -0700 | [diff] [blame] | 8 | "veyron.io/veyron/veyron/runtimes/google/testing/mocks/runtime" |
| 9 | "veyron.io/veyron/veyron/runtimes/google/vtrace" |
Matt Rosencrantz | 9fe6082 | 2014-09-12 10:09:53 -0700 | [diff] [blame] | 10 | |
Jiri Simsa | 519c507 | 2014-09-17 21:37:57 -0700 | [diff] [blame] | 11 | "veyron.io/veyron/veyron2/context" |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 12 | ) |
| 13 | |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 14 | // 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. |
| 18 | func testContext() context.T { |
Matt Rosencrantz | cc922c1 | 2014-11-28 20:28:59 -0800 | [diff] [blame] | 19 | ctx, _ := testContextWithoutDeadline().WithTimeout(20 * time.Second) |
| 20 | return ctx |
| 21 | } |
| 22 | |
| 23 | func testContextWithoutDeadline() context.T { |
Matt Rosencrantz | 9fe6082 | 2014-09-12 10:09:53 -0700 | [diff] [blame] | 24 | ctx := InternalNewContext(&runtime.PanicRuntime{}) |
Matt Rosencrantz | 3e76f28 | 2014-11-10 09:38:57 -0800 | [diff] [blame] | 25 | ctx, _ = vtrace.WithNewRootSpan(ctx, nil, false) |
Matt Rosencrantz | 9fe6082 | 2014-09-12 10:09:53 -0700 | [diff] [blame] | 26 | return ctx |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 27 | } |
| 28 | |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 29 | func 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 Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 56 | func 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 Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 77 | func TestCancelContext(t *testing.T) { |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 78 | ctx, cancel := testContext().WithCancel() |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 79 | testCancel(t, ctx, cancel) |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 80 | |
| 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 Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 94 | } |
| 95 | |
| 96 | func TestMultiLevelCancelContext(t *testing.T) { |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 97 | c0, c0Cancel := testContext().WithCancel() |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 98 | c1, _ := c0.WithCancel() |
| 99 | c2, _ := c1.WithCancel() |
| 100 | c3, _ := c2.WithCancel() |
| 101 | testCancel(t, c3, c0Cancel) |
| 102 | } |
| 103 | |
| 104 | type nonStandardContext struct { |
| 105 | context.T |
| 106 | } |
| 107 | |
| 108 | func (n *nonStandardContext) WithCancel() (ctx context.T, cancel context.CancelFunc) { |
| 109 | return newCancelContext(n) |
| 110 | } |
| 111 | func (n *nonStandardContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) { |
| 112 | return newDeadlineContext(n, deadline) |
| 113 | } |
| 114 | func (n *nonStandardContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) { |
| 115 | return newDeadlineContext(n, time.Now().Add(timeout)) |
| 116 | } |
| 117 | func (n *nonStandardContext) WithValue(key interface{}, val interface{}) context.T { |
| 118 | return newValueContext(n, key, val) |
| 119 | } |
| 120 | |
| 121 | func TestCancelContextWithNonStandard(t *testing.T) { |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 122 | // Test that cancellation flows properly through non-standard intermediates. |
| 123 | ctx := testContext() |
| 124 | c0 := &nonStandardContext{ctx} |
| 125 | c1, c1Cancel := c0.WithCancel() |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 126 | c2 := &nonStandardContext{c1} |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 127 | c3 := &nonStandardContext{c2} |
| 128 | c4, _ := c3.WithCancel() |
| 129 | testCancel(t, c4, c1Cancel) |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 130 | } |
| 131 | |
| 132 | func 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 | |
| 142 | func TestDeadlineContext(t *testing.T) { |
| 143 | cases := []time.Duration{ |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 144 | 3 * time.Millisecond, |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 145 | 0, |
| 146 | } |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 147 | rootCtx := InternalNewContext(&runtime.PanicRuntime{}) |
| 148 | cancelCtx, _ := rootCtx.WithCancel() |
| 149 | deadlineCtx, _ := rootCtx.WithDeadline(time.Now().Add(time.Hour)) |
| 150 | |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 151 | for _, desiredTimeout := range cases { |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 152 | // Test all the various ways of getting deadline contexts. |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 153 | start := time.Now() |
Matt Rosencrantz | 4d7c4be | 2014-11-12 16:23:33 -0800 | [diff] [blame] | 154 | 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 Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 175 | testDeadline(t, ctx, start, desiredTimeout) |
| 176 | } |
| 177 | |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 178 | ctx, cancel := testContext().WithDeadline(time.Now().Add(100 * time.Hour)) |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 179 | testCancel(t, ctx, cancel) |
| 180 | } |
| 181 | |
| 182 | func TestDeadlineContextWithRace(t *testing.T) { |
Matt Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 183 | ctx, cancel := testContext().WithDeadline(time.Now().Add(100 * time.Hour)) |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 184 | 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 | |
| 199 | func 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 Rosencrantz | bf85d54 | 2014-08-22 13:31:14 -0700 | [diff] [blame] | 212 | ctx1 := testContext().WithValue(key1, val1) |
Matt Rosencrantz | 137b8d2 | 2014-08-18 09:56:15 -0700 | [diff] [blame] | 213 | 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 | } |