Jiri Simsa | d7616c9 | 2015-03-24 23:44:30 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Vanadium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 5 | package manager |
| 6 | |
| 7 | import ( |
| 8 | "bytes" |
| 9 | "fmt" |
| 10 | "io" |
| 11 | "net" |
Cosmos Nicolaou | 920db00 | 2014-10-23 16:57:32 -0700 | [diff] [blame] | 12 | "os" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 13 | "reflect" |
| 14 | "runtime" |
Suharsh Sivakumar | ad1d419 | 2015-03-09 16:48:10 -0700 | [diff] [blame] | 15 | "sort" |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 16 | "strconv" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 17 | "strings" |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 18 | "syscall" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 19 | "testing" |
Cosmos Nicolaou | 920db00 | 2014-10-23 16:57:32 -0700 | [diff] [blame] | 20 | "time" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 21 | |
Cosmos Nicolaou | 1381f8a | 2015-03-13 09:40:34 -0700 | [diff] [blame] | 22 | "v.io/x/lib/vlog" |
| 23 | |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 24 | "v.io/v23/naming" |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 25 | "v.io/v23/rpc" |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 26 | "v.io/v23/security" |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 27 | |
Suharsh Sivakumar | dcc11d7 | 2015-05-11 12:19:20 -0700 | [diff] [blame] | 28 | inaming "v.io/x/ref/runtime/internal/naming" |
| 29 | _ "v.io/x/ref/runtime/internal/rpc/protocols/tcp" |
| 30 | _ "v.io/x/ref/runtime/internal/rpc/protocols/ws" |
| 31 | "v.io/x/ref/runtime/internal/rpc/stream" |
| 32 | "v.io/x/ref/runtime/internal/rpc/stream/vc" |
| 33 | "v.io/x/ref/runtime/internal/rpc/stream/vif" |
Cosmos Nicolaou | 1381f8a | 2015-03-13 09:40:34 -0700 | [diff] [blame] | 34 | "v.io/x/ref/test" |
| 35 | "v.io/x/ref/test/expect" |
| 36 | "v.io/x/ref/test/modules" |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 37 | "v.io/x/ref/test/testutil" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 38 | ) |
| 39 | |
Suharsh Sivakumar | d19c95d | 2015-02-19 14:44:50 -0800 | [diff] [blame] | 40 | // We write our own TestMain here instead of relying on v23 test generate because |
| 41 | // we need to set runtime.GOMAXPROCS. |
| 42 | func TestMain(m *testing.M) { |
Cosmos Nicolaou | 1381f8a | 2015-03-13 09:40:34 -0700 | [diff] [blame] | 43 | test.Init() |
Asim Shankar | c920db3 | 2014-10-16 19:18:21 -0700 | [diff] [blame] | 44 | // testutil.Init sets GOMAXPROCS to NumCPU. We want to force |
| 45 | // GOMAXPROCS to remain at 1, in order to trigger a particular race |
Suharsh Sivakumar | d19c95d | 2015-02-19 14:44:50 -0800 | [diff] [blame] | 46 | // condition that occurs when closing the server; also, using 1 cpu |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 47 | // introduces less variance in the behavior of the test. |
| 48 | runtime.GOMAXPROCS(1) |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 49 | modules.DispatchAndExitIfChild() |
Suharsh Sivakumar | d19c95d | 2015-02-19 14:44:50 -0800 | [diff] [blame] | 50 | os.Exit(m.Run()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 51 | } |
| 52 | |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 53 | func testSimpleFlow(t *testing.T, protocol string) { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 54 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 55 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 56 | pclient := testutil.NewPrincipal("client") |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 57 | pclient2 := testutil.NewPrincipal("client2") |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 58 | pserver := testutil.NewPrincipal("server") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 59 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 60 | ln, ep, err := server.Listen(protocol, "127.0.0.1:0", pserver, pserver.BlessingStore().Default()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 61 | if err != nil { |
| 62 | t.Fatal(err) |
| 63 | } |
| 64 | |
| 65 | data := "the dark knight rises" |
| 66 | var clientVC stream.VC |
| 67 | var clientF1 stream.Flow |
| 68 | go func() { |
Suharsh Sivakumar | 2ad4e10 | 2015-03-17 21:23:37 -0700 | [diff] [blame] | 69 | if clientVC, err = client.Dial(ep, pclient); err != nil { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 70 | t.Errorf("Dial(%q) failed: %v", ep, err) |
| 71 | return |
| 72 | } |
| 73 | if clientF1, err = clientVC.Connect(); err != nil { |
| 74 | t.Errorf("Connect() failed: %v", err) |
| 75 | return |
| 76 | } |
| 77 | if err := writeLine(clientF1, data); err != nil { |
| 78 | t.Error(err) |
| 79 | } |
| 80 | }() |
| 81 | serverF, err := ln.Accept() |
| 82 | if err != nil { |
| 83 | t.Fatalf("Accept failed: %v", err) |
| 84 | } |
| 85 | if got, err := readLine(serverF); got != data || err != nil { |
| 86 | t.Errorf("Got (%q, %v), want (%q, nil)", got, err, data) |
| 87 | } |
| 88 | // By this point, the goroutine has passed the write call (or exited |
| 89 | // early) since the read has gotten through. Check if the goroutine |
| 90 | // encountered any errors in creating the VC or flow and abort. |
| 91 | if t.Failed() { |
| 92 | return |
| 93 | } |
| 94 | defer clientF1.Close() |
| 95 | |
| 96 | ln.Close() |
| 97 | |
| 98 | // Writes on flows opened before the server listener was closed should |
| 99 | // still succeed. |
| 100 | data = "the dark knight goes to bed" |
| 101 | go func() { |
| 102 | if err := writeLine(clientF1, data); err != nil { |
| 103 | t.Error(err) |
| 104 | } |
| 105 | }() |
| 106 | if got, err := readLine(serverF); got != data || err != nil { |
| 107 | t.Errorf("Got (%q, %v), want (%q, nil)", got, err, data) |
| 108 | } |
| 109 | |
| 110 | // Opening a new flow on an existing VC will succeed initially, but |
| 111 | // writes on the client end will eventually fail once the server has |
| 112 | // stopped listening. |
| 113 | // |
| 114 | // It will require a round-trip to the server to notice the failure, |
| 115 | // hence the client should write enough data to ensure that the Write |
| 116 | // call will not return before a round-trip. |
| 117 | // |
| 118 | // The length of the data is taken to exceed the queue buffer size |
| 119 | // (DefaultBytesBufferedPerFlow), the shared counters (MaxSharedBytes) |
| 120 | // and the per-flow counters (DefaultBytesBufferedPerFlow) that are |
| 121 | // given when the flow gets established. |
| 122 | // |
| 123 | // TODO(caprita): separate the constants for the queue buffer size and |
| 124 | // the default number of counters to avoid confusion. |
| 125 | lotsOfData := string(make([]byte, vc.DefaultBytesBufferedPerFlow*2+vc.MaxSharedBytes+1)) |
| 126 | clientF2, err := clientVC.Connect() |
| 127 | if err != nil { |
| 128 | t.Fatalf("Connect() failed: %v", err) |
| 129 | } |
| 130 | defer clientF2.Close() |
| 131 | if err := writeLine(clientF2, lotsOfData); err == nil { |
| 132 | t.Errorf("Should not be able to Dial or Write after the Listener is closed") |
| 133 | } |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 134 | // Opening a new VC should fail fast. Note that we need to use a different |
| 135 | // principal since the client doesn't expect a response from a server |
| 136 | // when re-using VIF authentication. |
| 137 | if _, err := client.Dial(ep, pclient2); err == nil { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 138 | t.Errorf("Should not be able to Dial after listener is closed") |
| 139 | } |
| 140 | } |
| 141 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 142 | func TestSimpleFlow(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 143 | testSimpleFlow(t, "tcp") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 144 | } |
| 145 | |
| 146 | func TestSimpleFlowWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 147 | testSimpleFlow(t, "ws") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 148 | } |
| 149 | |
Cosmos Nicolaou | 9388ae4 | 2014-11-10 10:57:15 -0800 | [diff] [blame] | 150 | func TestConnectionTimeout(t *testing.T) { |
| 151 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
| 152 | |
| 153 | ch := make(chan error) |
| 154 | go func() { |
| 155 | // 203.0.113.0 is TEST-NET-3 from RFC5737 |
| 156 | ep, _ := inaming.NewEndpoint(naming.FormatEndpoint("tcp", "203.0.113.10:80")) |
Matt Rosencrantz | bf0d9d9 | 2015-04-08 12:43:14 -0700 | [diff] [blame] | 157 | _, err := client.Dial(ep, testutil.NewPrincipal("client"), DialTimeout(time.Second)) |
Cosmos Nicolaou | 9388ae4 | 2014-11-10 10:57:15 -0800 | [diff] [blame] | 158 | ch <- err |
| 159 | }() |
| 160 | |
| 161 | select { |
| 162 | case err := <-ch: |
| 163 | if err == nil { |
| 164 | t.Fatalf("expected an error") |
| 165 | } |
| 166 | case <-time.After(time.Minute): |
| 167 | t.Fatalf("timedout") |
| 168 | } |
| 169 | } |
| 170 | |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 171 | func testAuthenticatedByDefault(t *testing.T, protocol string) { |
Asim Shankar | 6b0510a | 2014-10-01 12:05:06 -0700 | [diff] [blame] | 172 | var ( |
| 173 | server = InternalNew(naming.FixedRoutingID(0x55555555)) |
| 174 | client = InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 175 | |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 176 | clientPrincipal = testutil.NewPrincipal("client") |
| 177 | serverPrincipal = testutil.NewPrincipal("server") |
Suharsh Sivakumar | 59c423c | 2015-03-11 14:06:03 -0700 | [diff] [blame] | 178 | clientKey = clientPrincipal.PublicKey() |
| 179 | serverBlessings = serverPrincipal.BlessingStore().Default() |
Asim Shankar | 6b0510a | 2014-10-01 12:05:06 -0700 | [diff] [blame] | 180 | ) |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 181 | ln, ep, err := server.Listen(protocol, "127.0.0.1:0", serverPrincipal, serverPrincipal.BlessingStore().Default()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 182 | if err != nil { |
| 183 | t.Fatal(err) |
| 184 | } |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 185 | // And the server blessing should be in the endpoint. |
| 186 | if got, want := ep.BlessingNames(), []string{"server"}; !reflect.DeepEqual(got, want) { |
| 187 | t.Errorf("Got blessings %v from endpoint, want %v", got, want) |
| 188 | } |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 189 | |
| 190 | errs := make(chan error) |
| 191 | |
Ankur | 50a5f39 | 2015-02-27 18:46:30 -0800 | [diff] [blame] | 192 | testAuth := func(tag string, flow stream.Flow, wantServer security.Blessings, wantClientKey security.PublicKey) { |
| 193 | // Since the client's blessing is expected to be self-signed we only test |
| 194 | // its public key |
| 195 | gotServer := flow.RemoteBlessings() |
| 196 | gotClientKey := flow.LocalBlessings().PublicKey() |
| 197 | if tag == "server" { |
| 198 | gotServer = flow.LocalBlessings() |
| 199 | gotClientKey = flow.RemoteBlessings().PublicKey() |
| 200 | } |
| 201 | if !reflect.DeepEqual(gotServer, wantServer) || !reflect.DeepEqual(gotClientKey, wantClientKey) { |
| 202 | errs <- fmt.Errorf("%s: Server: Got Blessings %q, want %q. Server: Got Blessings %q, want %q", tag, gotServer, wantServer, gotClientKey, wantClientKey) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 203 | return |
| 204 | } |
| 205 | errs <- nil |
| 206 | } |
| 207 | |
| 208 | go func() { |
| 209 | flow, err := ln.Accept() |
| 210 | if err != nil { |
| 211 | errs <- err |
| 212 | return |
| 213 | } |
| 214 | defer flow.Close() |
Ankur | 50a5f39 | 2015-02-27 18:46:30 -0800 | [diff] [blame] | 215 | testAuth("server", flow, serverBlessings, clientKey) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 216 | }() |
| 217 | |
| 218 | go func() { |
Suharsh Sivakumar | 2ad4e10 | 2015-03-17 21:23:37 -0700 | [diff] [blame] | 219 | vc, err := client.Dial(ep, clientPrincipal) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 220 | if err != nil { |
| 221 | errs <- err |
| 222 | return |
| 223 | } |
| 224 | flow, err := vc.Connect() |
| 225 | if err != nil { |
| 226 | errs <- err |
| 227 | return |
| 228 | } |
| 229 | defer flow.Close() |
Ankur | 50a5f39 | 2015-02-27 18:46:30 -0800 | [diff] [blame] | 230 | testAuth("client", flow, serverBlessings, clientKey) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 231 | }() |
| 232 | |
| 233 | if err := <-errs; err != nil { |
| 234 | t.Error(err) |
| 235 | } |
| 236 | if err := <-errs; err != nil { |
| 237 | t.Error(err) |
| 238 | } |
| 239 | } |
| 240 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 241 | func TestAuthenticatedByDefault(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 242 | testAuthenticatedByDefault(t, "tcp") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 243 | } |
| 244 | |
| 245 | func TestAuthenticatedByDefaultWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 246 | testAuthenticatedByDefault(t, "ws") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 247 | } |
| 248 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 249 | func numListeners(m stream.Manager) int { return len(m.(*manager).listeners) } |
| 250 | func debugString(m stream.Manager) string { return m.(*manager).DebugString() } |
| 251 | func numVIFs(m stream.Manager) int { return len(m.(*manager).vifs.List()) } |
| 252 | |
| 253 | func TestListenEndpoints(t *testing.T) { |
| 254 | server := InternalNew(naming.FixedRoutingID(0xcafe)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 255 | principal := testutil.NewPrincipal("test") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 256 | blessings := principal.BlessingStore().Default() |
| 257 | ln1, ep1, err1 := server.Listen("tcp", "127.0.0.1:0", principal, blessings) |
| 258 | ln2, ep2, err2 := server.Listen("tcp", "127.0.0.1:0", principal, blessings) |
Adam Sadovsky | 5181bdb | 2014-08-13 10:29:11 -0700 | [diff] [blame] | 259 | // Since "127.0.0.1:0" was used as the network address, a random port will be |
| 260 | // assigned in each case. The endpoint should include that random port. |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 261 | if err1 != nil { |
| 262 | t.Error(err1) |
| 263 | } |
| 264 | if err2 != nil { |
| 265 | t.Error(err2) |
| 266 | } |
| 267 | if ep1.String() == ep2.String() { |
| 268 | t.Errorf("Both listeners got the same endpoint: %q", ep1) |
| 269 | } |
| 270 | if n, expect := numListeners(server), 2; n != expect { |
| 271 | t.Errorf("expecting %d listeners, got %d for %s", n, expect, debugString(server)) |
| 272 | } |
| 273 | ln1.Close() |
| 274 | if n, expect := numListeners(server), 1; n != expect { |
| 275 | t.Errorf("expecting %d listeners, got %d for %s", n, expect, debugString(server)) |
| 276 | } |
| 277 | ln2.Close() |
| 278 | if n, expect := numListeners(server), 0; n != expect { |
| 279 | t.Errorf("expecting %d listeners, got %d for %s", n, expect, debugString(server)) |
| 280 | } |
| 281 | } |
| 282 | |
| 283 | func acceptLoop(ln stream.Listener) { |
| 284 | for { |
| 285 | f, err := ln.Accept() |
| 286 | if err != nil { |
| 287 | return |
| 288 | } |
| 289 | f.Close() |
| 290 | } |
| 291 | } |
| 292 | |
| 293 | func TestCloseListener(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 294 | testCloseListener(t, "tcp") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 295 | } |
| 296 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 297 | func TestCloseListenerWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 298 | testCloseListener(t, "ws") |
| 299 | } |
| 300 | |
| 301 | func testCloseListener(t *testing.T, protocol string) { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 302 | server := InternalNew(naming.FixedRoutingID(0x5e97e9)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 303 | pclient := testutil.NewPrincipal("client") |
| 304 | pserver := testutil.NewPrincipal("server") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 305 | blessings := pserver.BlessingStore().Default() |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 306 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 307 | ln, ep, err := server.Listen(protocol, "127.0.0.1:0", pserver, blessings) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 308 | if err != nil { |
| 309 | t.Fatal(err) |
| 310 | } |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 311 | // Server will just listen for flows and close them. |
| 312 | go acceptLoop(ln) |
| 313 | client := InternalNew(naming.FixedRoutingID(0xc1e41)) |
Suharsh Sivakumar | 2ad4e10 | 2015-03-17 21:23:37 -0700 | [diff] [blame] | 314 | if _, err = client.Dial(ep, pclient); err != nil { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 315 | t.Fatal(err) |
| 316 | } |
| 317 | ln.Close() |
| 318 | client = InternalNew(naming.FixedRoutingID(0xc1e42)) |
Suharsh Sivakumar | 2ad4e10 | 2015-03-17 21:23:37 -0700 | [diff] [blame] | 319 | if _, err := client.Dial(ep, pclient); err == nil { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 320 | t.Errorf("client.Dial(%q) should have failed", ep) |
| 321 | } |
| 322 | } |
| 323 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 324 | func TestShutdown(t *testing.T) { |
| 325 | server := InternalNew(naming.FixedRoutingID(0x5e97e9)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 326 | principal := testutil.NewPrincipal("test") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 327 | blessings := principal.BlessingStore().Default() |
| 328 | ln, _, err := server.Listen("tcp", "127.0.0.1:0", principal, blessings) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 329 | if err != nil { |
| 330 | t.Fatal(err) |
| 331 | } |
| 332 | // Server will just listen for flows and close them. |
| 333 | go acceptLoop(ln) |
| 334 | if n, expect := numListeners(server), 1; n != expect { |
| 335 | t.Errorf("expecting %d listeners, got %d for %s", n, expect, debugString(server)) |
| 336 | } |
| 337 | server.Shutdown() |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 338 | if _, _, err := server.Listen("tcp", "127.0.0.1:0", principal, blessings); err == nil { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 339 | t.Error("server should have shut down") |
| 340 | } |
| 341 | if n, expect := numListeners(server), 0; n != expect { |
| 342 | t.Errorf("expecting %d listeners, got %d for %s", n, expect, debugString(server)) |
| 343 | } |
| 344 | } |
| 345 | |
| 346 | func TestShutdownEndpoint(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 347 | testShutdownEndpoint(t, "tcp") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 348 | } |
| 349 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 350 | func TestShutdownEndpointWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 351 | testShutdownEndpoint(t, "ws") |
| 352 | } |
| 353 | |
| 354 | func testShutdownEndpoint(t *testing.T, protocol string) { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 355 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 356 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 357 | principal := testutil.NewPrincipal("test") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 358 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 359 | ln, ep, err := server.Listen(protocol, "127.0.0.1:0", principal, principal.BlessingStore().Default()) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 360 | if err != nil { |
| 361 | t.Fatal(err) |
| 362 | } |
| 363 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 364 | // Server will just listen for flows and close them. |
| 365 | go acceptLoop(ln) |
| 366 | |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 367 | vc, err := client.Dial(ep, testutil.NewPrincipal("client")) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 368 | if err != nil { |
| 369 | t.Fatal(err) |
| 370 | } |
| 371 | if f, err := vc.Connect(); f == nil || err != nil { |
| 372 | t.Errorf("vc.Connect failed: (%v, %v)", f, err) |
| 373 | } |
| 374 | client.ShutdownEndpoint(ep) |
| 375 | if f, err := vc.Connect(); f != nil || err == nil { |
| 376 | t.Errorf("vc.Connect unexpectedly succeeded: (%v, %v)", f, err) |
| 377 | } |
| 378 | } |
| 379 | |
Jungho Ahn | 6ab655f | 2015-04-14 18:27:09 -0700 | [diff] [blame] | 380 | func TestStartTimeout(t *testing.T) { |
| 381 | const ( |
| 382 | startTime = 5 * time.Millisecond |
Jungho Ahn | 6ab655f | 2015-04-14 18:27:09 -0700 | [diff] [blame] | 383 | ) |
| 384 | |
| 385 | var ( |
| 386 | server = InternalNew(naming.FixedRoutingID(0x55555555)) |
| 387 | pserver = testutil.NewPrincipal("server") |
| 388 | lopts = []stream.ListenerOpt{vc.StartTimeout{startTime}} |
| 389 | ) |
| 390 | |
| 391 | // Pause the start timers. |
| 392 | triggerTimers := vif.SetFakeTimers() |
| 393 | |
| 394 | ln, ep, err := server.Listen("tcp", "127.0.0.1:0", pserver, pserver.BlessingStore().Default(), lopts...) |
| 395 | if err != nil { |
| 396 | t.Fatal(err) |
| 397 | } |
| 398 | go func() { |
| 399 | for { |
| 400 | _, err := ln.Accept() |
| 401 | if err != nil { |
| 402 | return |
| 403 | } |
| 404 | } |
| 405 | }() |
| 406 | |
| 407 | _, err = net.Dial(ep.Addr().Network(), ep.Addr().String()) |
| 408 | if err != nil { |
| 409 | t.Fatalf("net.Dial failed: %v", err) |
| 410 | } |
| 411 | |
| 412 | // Trigger the start timers. |
| 413 | triggerTimers() |
| 414 | |
| 415 | // No VC is opened. The VIF should be closed after start timeout. |
Jungho Ahn | cc9d572 | 2015-04-22 10:13:04 -0700 | [diff] [blame] | 416 | for range time.Tick(startTime) { |
| 417 | if numVIFs(server) == 0 { |
| 418 | break |
Jungho Ahn | 6ab655f | 2015-04-14 18:27:09 -0700 | [diff] [blame] | 419 | } |
| 420 | } |
Jungho Ahn | 6ab655f | 2015-04-14 18:27:09 -0700 | [diff] [blame] | 421 | } |
| 422 | |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 423 | func testIdleTimeout(t *testing.T, testServer bool) { |
| 424 | const ( |
| 425 | idleTime = 10 * time.Millisecond |
| 426 | // We use a long wait time here since it takes some time to handle VC close |
| 427 | // especially in race testing. |
| 428 | waitTime = 150 * time.Millisecond |
| 429 | ) |
| 430 | |
| 431 | var ( |
| 432 | server = InternalNew(naming.FixedRoutingID(0x55555555)) |
| 433 | client = InternalNew(naming.FixedRoutingID(0xcccccccc)) |
| 434 | pclient = testutil.NewPrincipal("client") |
| 435 | pserver = testutil.NewPrincipal("server") |
| 436 | |
| 437 | opts []stream.VCOpt |
| 438 | lopts []stream.ListenerOpt |
| 439 | ) |
| 440 | if testServer { |
| 441 | lopts = []stream.ListenerOpt{vc.IdleTimeout{idleTime}} |
| 442 | } else { |
| 443 | opts = []stream.VCOpt{vc.IdleTimeout{idleTime}} |
| 444 | } |
| 445 | |
| 446 | // Pause the idle timers. |
| 447 | triggerTimers := vif.SetFakeTimers() |
| 448 | |
| 449 | ln, ep, err := server.Listen("tcp", "127.0.0.1:0", pserver, pserver.BlessingStore().Default(), lopts...) |
| 450 | if err != nil { |
| 451 | t.Fatal(err) |
| 452 | } |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 453 | errch := make(chan error) |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 454 | go func() { |
| 455 | for { |
| 456 | _, err := ln.Accept() |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 457 | errch <- err |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 458 | } |
| 459 | }() |
| 460 | |
| 461 | vc, err := client.Dial(ep, pclient, opts...) |
| 462 | if err != nil { |
| 463 | t.Fatalf("client.Dial(%q) failed: %v", ep, err) |
| 464 | } |
| 465 | f, err := vc.Connect() |
| 466 | if f == nil || err != nil { |
| 467 | t.Fatalf("vc.Connect failed: (%v, %v)", f, err) |
| 468 | } |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 469 | // Wait until the server accepts the flow or fails. |
| 470 | if err = <-errch; err != nil { |
| 471 | t.Fatalf("ln.Accept failed: %v", err) |
| 472 | } |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 473 | |
| 474 | // Trigger the idle timers. |
| 475 | triggerTimers() |
| 476 | |
| 477 | // One active flow. The VIF should be kept open. |
| 478 | time.Sleep(waitTime) |
| 479 | if n := numVIFs(client); n != 1 { |
| 480 | t.Errorf("Client has %d VIFs; want 1\n%v", n, debugString(client)) |
| 481 | } |
| 482 | if n := numVIFs(server); n != 1 { |
| 483 | t.Errorf("Server has %d VIFs; want 1\n%v", n, debugString(server)) |
| 484 | } |
| 485 | |
| 486 | f.Close() |
| 487 | |
| 488 | // The flow has been closed. The VIF should be closed after idle timeout. |
Jungho Ahn | cc9d572 | 2015-04-22 10:13:04 -0700 | [diff] [blame] | 489 | for range time.Tick(idleTime) { |
| 490 | if numVIFs(client) == 0 && numVIFs(server) == 0 { |
| 491 | break |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 492 | } |
| 493 | } |
Jungho Ahn | cd175b8 | 2015-03-27 14:29:40 -0700 | [diff] [blame] | 494 | } |
| 495 | |
| 496 | func TestIdleTimeout(t *testing.T) { testIdleTimeout(t, false) } |
| 497 | func TestIdleTimeoutServer(t *testing.T) { testIdleTimeout(t, true) } |
| 498 | |
Andres Erbsen | ffa4574 | 2014-08-13 10:13:11 -0700 | [diff] [blame] | 499 | /* TLS + resumption + channel bindings is broken: <https://secure-resumption.com/#channelbindings>. |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 500 | func TestSessionTicketCache(t *testing.T) { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 501 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 502 | _, ep, err := server.Listen("tcp", "127.0.0.1:0", testutil.NewPrincipal("server")) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 503 | if err != nil { |
| 504 | t.Fatal(err) |
| 505 | } |
| 506 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 507 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 508 | if _, err = client.Dial(ep, testutil.NewPrincipal("TestSessionTicketCacheClient")); err != nil { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 509 | t.Fatalf("Dial(%q) failed: %v", ep, err) |
| 510 | } |
| 511 | |
| 512 | if _, ok := client.(*manager).sessionCache.Get(ep.String()); !ok { |
| 513 | t.Fatalf("SessionTicket from TLS handshake not cached") |
| 514 | } |
| 515 | } |
Andres Erbsen | ffa4574 | 2014-08-13 10:13:11 -0700 | [diff] [blame] | 516 | */ |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 517 | |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 518 | func testMultipleVCs(t *testing.T, protocol string) { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 519 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 520 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 521 | principal := testutil.NewPrincipal("test") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 522 | |
| 523 | const nVCs = 2 |
| 524 | const data = "bugs bunny" |
| 525 | |
| 526 | // Have the server read from each flow and write to rchan. |
| 527 | rchan := make(chan string) |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 528 | ln, ep, err := server.Listen(protocol, "127.0.0.1:0", principal, principal.BlessingStore().Default()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 529 | if err != nil { |
| 530 | t.Fatal(err) |
| 531 | } |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 532 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 533 | read := func(flow stream.Flow, c chan string) { |
| 534 | var buf bytes.Buffer |
| 535 | var tmp [1024]byte |
| 536 | for { |
| 537 | n, err := flow.Read(tmp[:]) |
| 538 | buf.Write(tmp[:n]) |
| 539 | if err == io.EOF { |
| 540 | c <- buf.String() |
| 541 | return |
| 542 | } |
| 543 | if err != nil { |
| 544 | t.Error(err) |
| 545 | return |
| 546 | } |
| 547 | } |
| 548 | } |
| 549 | go func() { |
| 550 | for i := 0; i < nVCs; i++ { |
| 551 | flow, err := ln.Accept() |
| 552 | if err != nil { |
| 553 | t.Error(err) |
| 554 | rchan <- "" |
| 555 | continue |
| 556 | } |
| 557 | go read(flow, rchan) |
| 558 | } |
| 559 | }() |
| 560 | |
| 561 | // Have the client establish nVCs and a flow on each. |
| 562 | var vcs [nVCs]stream.VC |
| 563 | for i := 0; i < nVCs; i++ { |
| 564 | var err error |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 565 | vcs[i], err = client.Dial(ep, testutil.NewPrincipal("client")) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 566 | if err != nil { |
| 567 | t.Fatal(err) |
| 568 | } |
| 569 | } |
| 570 | write := func(vc stream.VC) { |
| 571 | if err != nil { |
| 572 | ln.Close() |
| 573 | t.Error(err) |
| 574 | return |
| 575 | } |
| 576 | flow, err := vc.Connect() |
| 577 | if err != nil { |
| 578 | ln.Close() |
| 579 | t.Error(err) |
| 580 | return |
| 581 | } |
| 582 | defer flow.Close() |
| 583 | if _, err := flow.Write([]byte(data)); err != nil { |
| 584 | ln.Close() |
| 585 | t.Error(err) |
| 586 | return |
| 587 | } |
| 588 | } |
| 589 | for _, vc := range vcs { |
| 590 | go write(vc) |
| 591 | } |
| 592 | for i := 0; i < nVCs; i++ { |
| 593 | if got := <-rchan; got != data { |
| 594 | t.Errorf("Got %q want %q", got, data) |
| 595 | } |
| 596 | } |
| 597 | } |
| 598 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 599 | func TestMultipleVCs(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 600 | testMultipleVCs(t, "tcp") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 601 | } |
| 602 | |
| 603 | func TestMultipleVCsWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 604 | testMultipleVCs(t, "ws") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 605 | } |
| 606 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 607 | func TestAddressResolution(t *testing.T) { |
| 608 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 609 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 610 | principal := testutil.NewPrincipal("test") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 611 | |
Adam Sadovsky | 5181bdb | 2014-08-13 10:29:11 -0700 | [diff] [blame] | 612 | // Using "tcp4" instead of "tcp" because the latter can end up with IPv6 |
| 613 | // addresses and our Google Compute Engine integration test machines cannot |
| 614 | // resolve IPv6 addresses. |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 615 | // As of April 2014, https://developers.google.com/compute/docs/networking |
| 616 | // said that IPv6 is not yet supported. |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 617 | ln, ep, err := server.Listen("tcp4", "127.0.0.1:0", principal, principal.BlessingStore().Default()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 618 | if err != nil { |
| 619 | t.Fatal(err) |
| 620 | } |
| 621 | go acceptLoop(ln) |
| 622 | |
Adam Sadovsky | 5181bdb | 2014-08-13 10:29:11 -0700 | [diff] [blame] | 623 | // We'd like an endpoint that contains an address that's different than the |
| 624 | // one used for the connection. In practice this is awkward to achieve since |
| 625 | // we don't want to listen on ":0" since that will annoy firewalls. Instead we |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 626 | // create a endpoint with "localhost", which will result in an endpoint that |
| 627 | // doesn't contain 127.0.0.1. |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 628 | _, port, _ := net.SplitHostPort(ep.Addr().String()) |
Matt Rosencrantz | c16339c | 2015-04-23 10:47:06 -0700 | [diff] [blame] | 629 | nep := &inaming.Endpoint{ |
| 630 | Protocol: ep.Addr().Network(), |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 631 | Address: net.JoinHostPort("localhost", port), |
Matt Rosencrantz | c16339c | 2015-04-23 10:47:06 -0700 | [diff] [blame] | 632 | RID: ep.RoutingID(), |
| 633 | } |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 634 | |
| 635 | // Dial multiple VCs |
| 636 | for i := 0; i < 2; i++ { |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 637 | if _, err = client.Dial(nep, testutil.NewPrincipal("client")); err != nil { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 638 | t.Fatalf("Dial #%d failed: %v", i, err) |
| 639 | } |
| 640 | } |
| 641 | // They should all be on the same VIF. |
| 642 | if n := numVIFs(client); n != 1 { |
| 643 | t.Errorf("Client has %d VIFs, want 1\n%v", n, debugString(client)) |
| 644 | } |
| 645 | // TODO(ashankar): While a VIF can be re-used to Dial from the server |
| 646 | // to the client, currently there is no way to have the client "listen" |
| 647 | // on the same VIF. It can listen on a VC for new flows, but it cannot |
| 648 | // listen on an established VIF for new VCs. Figure this out? |
| 649 | } |
| 650 | |
| 651 | func TestServerRestartDuringClientLifetime(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 652 | testServerRestartDuringClientLifetime(t, "tcp") |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 653 | } |
| 654 | |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 655 | func TestServerRestartDuringClientLifetimeWS(t *testing.T) { |
Cosmos Nicolaou | ae8dd21 | 2014-12-13 23:43:08 -0800 | [diff] [blame] | 656 | testServerRestartDuringClientLifetime(t, "ws") |
| 657 | } |
| 658 | |
| 659 | func testServerRestartDuringClientLifetime(t *testing.T, protocol string) { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 660 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 661 | pclient := testutil.NewPrincipal("client") |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 662 | pclient2 := testutil.NewPrincipal("client2") |
Cosmos Nicolaou | 9e90984 | 2015-03-17 11:58:59 -0700 | [diff] [blame] | 663 | sh, err := modules.NewShell(nil, nil, testing.Verbose(), t) |
Cosmos Nicolaou | 344cc4a | 2014-11-26 15:38:43 -0800 | [diff] [blame] | 664 | if err != nil { |
| 665 | t.Fatalf("unexpected error: %s", err) |
| 666 | } |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 667 | defer sh.Cleanup(nil, nil) |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 668 | h, err := sh.Start(nil, runServer, protocol, "127.0.0.1:0") |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 669 | if err != nil { |
| 670 | t.Fatalf("unexpected error: %s", err) |
| 671 | } |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 672 | epstr := expect.NewSession(t, h.Stdout(), time.Minute).ExpectVar("ENDPOINT") |
| 673 | ep, err := inaming.NewEndpoint(epstr) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 674 | if err != nil { |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 675 | t.Fatalf("inaming.NewEndpoint(%q): %v", epstr, err) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 676 | } |
Suharsh Sivakumar | 2ad4e10 | 2015-03-17 21:23:37 -0700 | [diff] [blame] | 677 | if _, err := client.Dial(ep, pclient); err != nil { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 678 | t.Fatal(err) |
| 679 | } |
| 680 | h.Shutdown(nil, os.Stderr) |
| 681 | |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 682 | // A new VC cannot be created since the server is dead. Note that we need to |
| 683 | // use a different principal since the client doesn't expect a response from |
| 684 | // a server when re-using VIF authentication. |
| 685 | if _, err := client.Dial(ep, pclient2); err == nil { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 686 | t.Fatal("Expected client.Dial to fail since server is dead") |
| 687 | } |
| 688 | |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 689 | h, err = sh.Start(nil, runServer, protocol, ep.Addr().String()) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 690 | if err != nil { |
| 691 | t.Fatalf("unexpected error: %s", err) |
| 692 | } |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 693 | // Restarting the server, listening on the same address as before |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 694 | ep2, err := inaming.NewEndpoint(expect.NewSession(t, h.Stdout(), time.Minute).ExpectVar("ENDPOINT")) |
| 695 | if err != nil { |
| 696 | t.Fatal(err) |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 697 | } |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 698 | if got, want := ep.Addr().String(), ep2.Addr().String(); got != want { |
| 699 | t.Fatalf("Got %q, want %q", got, want) |
| 700 | } |
| 701 | if _, err := client.Dial(ep2, pclient); err != nil { |
Shyam Jayaraman | dbae76b | 2014-11-17 12:51:29 -0800 | [diff] [blame] | 702 | t.Fatal(err) |
| 703 | } |
| 704 | } |
| 705 | |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 706 | var runServer = modules.Register(runServerFunc, "runServer") |
| 707 | |
| 708 | func runServerFunc(env *modules.Env, args ...string) error { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 709 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 710 | principal := testutil.NewPrincipal("test") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 711 | _, ep, err := server.Listen(args[0], args[1], principal, principal.BlessingStore().Default()) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 712 | if err != nil { |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 713 | fmt.Fprintln(env.Stderr, err) |
Cosmos Nicolaou | 920db00 | 2014-10-23 16:57:32 -0700 | [diff] [blame] | 714 | return err |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 715 | } |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 716 | fmt.Fprintf(env.Stdout, "ENDPOINT=%v\n", ep) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 717 | // Live forever (till the process is explicitly killed) |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 718 | modules.WaitForEOF(env.Stdin) |
Cosmos Nicolaou | 920db00 | 2014-10-23 16:57:32 -0700 | [diff] [blame] | 719 | return nil |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 720 | } |
| 721 | |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 722 | var runRLimitedServer = modules.Register(func(env *modules.Env, args ...string) error { |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 723 | var rlimit syscall.Rlimit |
| 724 | if err := syscall.Getrlimit(syscall.RLIMIT_NOFILE, &rlimit); err != nil { |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 725 | fmt.Fprintln(env.Stderr, err) |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 726 | return err |
| 727 | } |
| 728 | rlimit.Cur = 9 |
| 729 | if err := syscall.Setrlimit(syscall.RLIMIT_NOFILE, &rlimit); err != nil { |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 730 | fmt.Fprintln(env.Stderr, err) |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 731 | return err |
| 732 | } |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 733 | fmt.Fprintf(env.Stdout, "RLIMIT_NOFILE=%d\n", rlimit.Cur) |
| 734 | return runServerFunc(env, args...) |
| 735 | }, "runRLimitedServer") |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 736 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 737 | func readLine(f stream.Flow) (string, error) { |
| 738 | var result bytes.Buffer |
| 739 | var buf [5]byte |
| 740 | for { |
| 741 | n, err := f.Read(buf[:]) |
| 742 | result.Write(buf[:n]) |
| 743 | if err == io.EOF || buf[n-1] == '\n' { |
| 744 | return strings.TrimRight(result.String(), "\n"), nil |
| 745 | } |
| 746 | if err != nil { |
| 747 | return "", fmt.Errorf("Read returned (%d, %v)", n, err) |
| 748 | } |
| 749 | } |
| 750 | } |
| 751 | |
| 752 | func writeLine(f stream.Flow, data string) error { |
| 753 | data = data + "\n" |
| 754 | vlog.VI(1).Infof("write sending %d bytes", len(data)) |
| 755 | if n, err := f.Write([]byte(data)); err != nil { |
| 756 | return fmt.Errorf("Write returned (%d, %v)", n, err) |
| 757 | } |
| 758 | return nil |
| 759 | } |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 760 | |
| 761 | func TestRegistration(t *testing.T) { |
| 762 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 763 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 764 | principal := testutil.NewPrincipal("server") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 765 | blessings := principal.BlessingStore().Default() |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 766 | |
Cosmos Nicolaou | 87c0a55 | 2014-12-02 23:05:49 -0800 | [diff] [blame] | 767 | dialer := func(_, _ string, _ time.Duration) (net.Conn, error) { |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 768 | return nil, fmt.Errorf("tn.Dial") |
| 769 | } |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 770 | resolver := func(_, _ string) (string, string, error) { |
| 771 | return "", "", fmt.Errorf("tn.Resolve") |
| 772 | } |
Cosmos Nicolaou | 87c0a55 | 2014-12-02 23:05:49 -0800 | [diff] [blame] | 773 | listener := func(_, _ string) (net.Listener, error) { |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 774 | return nil, fmt.Errorf("tn.Listen") |
| 775 | } |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 776 | rpc.RegisterProtocol("tn", dialer, resolver, listener) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 777 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 778 | _, _, err := server.Listen("tnx", "127.0.0.1:0", principal, blessings) |
Cosmos Nicolaou | 07a736d | 2015-04-06 11:55:26 -0700 | [diff] [blame] | 779 | if err == nil || !strings.Contains(err.Error(), "unknown network: tnx") { |
| 780 | t.Fatalf("expected error is missing (%v)", err) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 781 | } |
| 782 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 783 | _, _, err = server.Listen("tn", "127.0.0.1:0", principal, blessings) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 784 | if err == nil || !strings.Contains(err.Error(), "tn.Listen") { |
Cosmos Nicolaou | 07a736d | 2015-04-06 11:55:26 -0700 | [diff] [blame] | 785 | t.Fatalf("expected error is missing (%v)", err) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 786 | } |
| 787 | |
| 788 | // Need a functional listener to test Dial. |
Cosmos Nicolaou | 87c0a55 | 2014-12-02 23:05:49 -0800 | [diff] [blame] | 789 | listener = func(_, addr string) (net.Listener, error) { |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 790 | return net.Listen("tcp", addr) |
| 791 | } |
| 792 | |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 793 | if got, want := rpc.RegisterProtocol("tn", dialer, resolver, listener), true; got != want { |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 794 | t.Errorf("got %t, want %t", got, want) |
| 795 | } |
| 796 | |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 797 | _, ep, err := server.Listen("tn", "127.0.0.1:0", principal, blessings) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 798 | if err != nil { |
| 799 | t.Errorf("unexpected error %s", err) |
| 800 | } |
| 801 | |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 802 | _, err = client.Dial(ep, testutil.NewPrincipal("client")) |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 803 | if err == nil || !strings.Contains(err.Error(), "tn.Resolve") { |
| 804 | t.Fatalf("expected error is missing (%v)", err) |
Cosmos Nicolaou | 5129fcb | 2014-08-22 11:52:25 -0700 | [diff] [blame] | 805 | } |
| 806 | } |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 807 | |
| 808 | func TestBlessingNamesInEndpoint(t *testing.T) { |
| 809 | var ( |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 810 | p = testutil.NewPrincipal("default") |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 811 | b, _ = p.BlessSelf("dev.v.io/users/foo@bar.com/devices/desktop/app/myapp") |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 812 | |
| 813 | server = InternalNew(naming.FixedRoutingID(0x1)) |
| 814 | |
| 815 | tests = []struct { |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 816 | principal security.Principal |
| 817 | blessings security.Blessings |
| 818 | blessingNames []string |
| 819 | err bool |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 820 | }{ |
| 821 | { |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 822 | // provided blessings should match returned output. |
| 823 | principal: p, |
| 824 | blessings: b, |
| 825 | blessingNames: []string{"dev.v.io/users/foo@bar.com/devices/desktop/app/myapp"}, |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 826 | }, |
| 827 | { |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 828 | // It is an error to provide a principal without providing blessings. |
Suharsh Sivakumar | 59c423c | 2015-03-11 14:06:03 -0700 | [diff] [blame] | 829 | principal: p, |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 830 | blessings: security.Blessings{}, |
Suharsh Sivakumar | 59c423c | 2015-03-11 14:06:03 -0700 | [diff] [blame] | 831 | err: true, |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 832 | }, |
| 833 | { |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 834 | // It is an error to provide inconsistent blessings and principal |
Asim Shankar | 4a69828 | 2015-03-21 21:59:18 -0700 | [diff] [blame] | 835 | principal: testutil.NewPrincipal("random"), |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 836 | blessings: b, |
Suharsh Sivakumar | 59c423c | 2015-03-11 14:06:03 -0700 | [diff] [blame] | 837 | err: true, |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 838 | }, |
| 839 | } |
| 840 | ) |
| 841 | // p must recognize its own blessings! |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 842 | p.AddToRoots(b) |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 843 | for idx, test := range tests { |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 844 | ln, ep, err := server.Listen("tcp", "127.0.0.1:0", test.principal, test.blessings) |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 845 | if (err != nil) != test.err { |
| 846 | t.Errorf("test #%d: Got error %v, wanted error: %v", idx, err, test.err) |
| 847 | } |
| 848 | if err != nil { |
| 849 | continue |
| 850 | } |
| 851 | ln.Close() |
Suharsh Sivakumar | e5e5dcc | 2015-03-18 14:29:31 -0700 | [diff] [blame] | 852 | got, want := ep.BlessingNames(), test.blessingNames |
Suharsh Sivakumar | ad1d419 | 2015-03-09 16:48:10 -0700 | [diff] [blame] | 853 | sort.Strings(got) |
| 854 | sort.Strings(want) |
| 855 | if !reflect.DeepEqual(got, want) { |
Asim Shankar | 7171a25 | 2015-03-07 14:41:40 -0800 | [diff] [blame] | 856 | t.Errorf("test #%d: Got %v, want %v", idx, got, want) |
| 857 | } |
| 858 | } |
| 859 | } |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 860 | |
| 861 | func TestVIFCleanupWhenFDLimitIsReached(t *testing.T) { |
| 862 | sh, err := modules.NewShell(nil, nil, testing.Verbose(), t) |
| 863 | if err != nil { |
| 864 | t.Fatal(err) |
| 865 | } |
| 866 | defer sh.Cleanup(nil, nil) |
Todd Wang | 9587390 | 2015-05-22 14:21:30 -0700 | [diff] [blame] | 867 | h, err := sh.Start(nil, runRLimitedServer, "--logtostderr=true", "tcp", "127.0.0.1:0") |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 868 | if err != nil { |
| 869 | t.Fatal(err) |
| 870 | } |
| 871 | defer h.CloseStdin() |
| 872 | stdout := expect.NewSession(t, h.Stdout(), time.Minute) |
| 873 | nfiles, err := strconv.Atoi(stdout.ExpectVar("RLIMIT_NOFILE")) |
| 874 | if stdout.Error() != nil { |
| 875 | t.Fatal(stdout.Error()) |
| 876 | } |
| 877 | if err != nil { |
| 878 | t.Fatal(err) |
| 879 | } |
| 880 | epstr := stdout.ExpectVar("ENDPOINT") |
| 881 | if stdout.Error() != nil { |
| 882 | t.Fatal(stdout.Error()) |
| 883 | } |
| 884 | ep, err := inaming.NewEndpoint(epstr) |
| 885 | if err != nil { |
| 886 | t.Fatal(err) |
| 887 | } |
| 888 | // Different client processes (represented by different stream managers |
| 889 | // in this test) should be able to make progress, even if the server |
| 890 | // has reached its file descriptor limit. |
| 891 | nattempts := 0 |
| 892 | for i := 0; i < 2*nfiles; i++ { |
| 893 | client := InternalNew(naming.FixedRoutingID(uint64(i))) |
| 894 | defer client.Shutdown() |
| 895 | principal := testutil.NewPrincipal(fmt.Sprintf("client%d", i)) |
| 896 | connected := false |
| 897 | for !connected { |
| 898 | nattempts++ |
| 899 | // If the client connection reached the server when it |
| 900 | // was at its limit, it might fail. However, this |
| 901 | // failure will trigger the "kill connections" logic at |
| 902 | // the server and eventually the client should succeed. |
| 903 | vc, err := client.Dial(ep, principal) |
| 904 | if err != nil { |
| 905 | continue |
| 906 | } |
| 907 | // Establish a flow to prevent the VC (and thus the |
| 908 | // underlying VIF) from being garbage collected as an |
| 909 | // "inactive" connection. |
| 910 | flow, err := vc.Connect() |
| 911 | if err != nil { |
| 912 | continue |
| 913 | } |
| 914 | defer flow.Close() |
| 915 | connected = true |
| 916 | } |
| 917 | } |
| 918 | var stderr bytes.Buffer |
| 919 | if err := h.Shutdown(nil, &stderr); err != nil { |
Cosmos Nicolaou | c818b80 | 2015-06-05 15:52:45 -0700 | [diff] [blame] | 920 | t.Logf("%s", stderr.String()) |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 921 | t.Fatal(err) |
| 922 | } |
Matt Rosencrantz | a40f12a | 2015-04-24 11:07:55 -0700 | [diff] [blame] | 923 | if log := expect.NewSession(t, bytes.NewReader(stderr.Bytes()), time.Minute).ExpectSetEventuallyRE("listener.go.*Killing [1-9][0-9]* Conns"); len(log) == 0 { |
| 924 | t.Errorf("Failed to find log message talking about killing Conns in:\n%v", stderr.String()) |
Asim Shankar | e89936f | 2015-04-22 17:37:43 -0700 | [diff] [blame] | 925 | } |
| 926 | t.Logf("Server FD limit:%d", nfiles) |
| 927 | t.Logf("Client connection attempts: %d", nattempts) |
| 928 | } |
Suharsh Sivakumar | 859ea0f | 2015-04-29 23:51:39 -0700 | [diff] [blame] | 929 | |
| 930 | func TestConcurrentDials(t *testing.T) { |
| 931 | // Concurrent Dials to the same network, address should only result in one VIF. |
| 932 | server := InternalNew(naming.FixedRoutingID(0x55555555)) |
| 933 | client := InternalNew(naming.FixedRoutingID(0xcccccccc)) |
| 934 | principal := testutil.NewPrincipal("test") |
| 935 | |
| 936 | // Using "tcp4" instead of "tcp" because the latter can end up with IPv6 |
| 937 | // addresses and our Google Compute Engine integration test machines cannot |
| 938 | // resolve IPv6 addresses. |
| 939 | // As of April 2014, https://developers.google.com/compute/docs/networking |
| 940 | // said that IPv6 is not yet supported. |
| 941 | ln, ep, err := server.Listen("tcp4", "127.0.0.1:0", principal, principal.BlessingStore().Default()) |
| 942 | if err != nil { |
| 943 | t.Fatal(err) |
| 944 | } |
| 945 | go acceptLoop(ln) |
| 946 | |
Suharsh Sivakumar | 859ea0f | 2015-04-29 23:51:39 -0700 | [diff] [blame] | 947 | nep := &inaming.Endpoint{ |
| 948 | Protocol: ep.Addr().Network(), |
Suharsh Sivakumar | 7e93ce5 | 2015-05-07 17:46:13 -0700 | [diff] [blame] | 949 | Address: ep.Addr().String(), |
Suharsh Sivakumar | 859ea0f | 2015-04-29 23:51:39 -0700 | [diff] [blame] | 950 | RID: ep.RoutingID(), |
| 951 | } |
| 952 | |
| 953 | // Dial multiple VCs |
| 954 | errCh := make(chan error, 10) |
| 955 | for i := 0; i < 10; i++ { |
| 956 | go func() { |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 957 | _, err := client.Dial(nep, testutil.NewPrincipal("client")) |
Suharsh Sivakumar | 859ea0f | 2015-04-29 23:51:39 -0700 | [diff] [blame] | 958 | errCh <- err |
| 959 | }() |
| 960 | } |
| 961 | for i := 0; i < 10; i++ { |
Jungho Ahn | 19e84b2 | 2015-05-18 13:22:27 -0700 | [diff] [blame] | 962 | if err := <-errCh; err != nil { |
Suharsh Sivakumar | 859ea0f | 2015-04-29 23:51:39 -0700 | [diff] [blame] | 963 | t.Fatal(err) |
| 964 | } |
| 965 | } |
| 966 | // They should all be on the same VIF. |
| 967 | if n := numVIFs(client); n != 1 { |
| 968 | t.Errorf("Client has %d VIFs, want 1\n%v", n, debugString(client)) |
| 969 | } |
| 970 | } |