blob: 6fe2ddd4324cca98d6646a17c676154ea7972179 [file] [log] [blame]
Jiri Simsad7616c92015-03-24 23:44:30 -07001// 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 Simsa5293dcb2014-05-10 09:56:38 -07005package manager
6
7import (
8 "bytes"
9 "fmt"
10 "io"
11 "net"
Cosmos Nicolaou920db002014-10-23 16:57:32 -070012 "os"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070013 "reflect"
14 "runtime"
Suharsh Sivakumarad1d4192015-03-09 16:48:10 -070015 "sort"
Asim Shankare89936f2015-04-22 17:37:43 -070016 "strconv"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070017 "strings"
Asim Shankare89936f2015-04-22 17:37:43 -070018 "syscall"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070019 "testing"
Cosmos Nicolaou920db002014-10-23 16:57:32 -070020 "time"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070021
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070022 "v.io/x/lib/vlog"
23
Jiri Simsa6ac95222015-02-23 16:11:49 -080024 "v.io/v23/naming"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070025 "v.io/v23/rpc"
Jiri Simsa6ac95222015-02-23 16:11:49 -080026 "v.io/v23/security"
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -070027
Suharsh Sivakumardcc11d72015-05-11 12:19:20 -070028 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 Nicolaou1381f8a2015-03-13 09:40:34 -070034 "v.io/x/ref/test"
35 "v.io/x/ref/test/expect"
36 "v.io/x/ref/test/modules"
Asim Shankar4a698282015-03-21 21:59:18 -070037 "v.io/x/ref/test/testutil"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070038)
39
Suharsh Sivakumard19c95d2015-02-19 14:44:50 -080040// We write our own TestMain here instead of relying on v23 test generate because
41// we need to set runtime.GOMAXPROCS.
42func TestMain(m *testing.M) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070043 test.Init()
Asim Shankarc920db32014-10-16 19:18:21 -070044 // testutil.Init sets GOMAXPROCS to NumCPU. We want to force
45 // GOMAXPROCS to remain at 1, in order to trigger a particular race
Suharsh Sivakumard19c95d2015-02-19 14:44:50 -080046 // condition that occurs when closing the server; also, using 1 cpu
Jiri Simsa5293dcb2014-05-10 09:56:38 -070047 // introduces less variance in the behavior of the test.
48 runtime.GOMAXPROCS(1)
Todd Wang95873902015-05-22 14:21:30 -070049 modules.DispatchAndExitIfChild()
Suharsh Sivakumard19c95d2015-02-19 14:44:50 -080050 os.Exit(m.Run())
Jiri Simsa5293dcb2014-05-10 09:56:38 -070051}
52
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -080053func testSimpleFlow(t *testing.T, protocol string) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070054 server := InternalNew(naming.FixedRoutingID(0x55555555))
55 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -070056 pclient := testutil.NewPrincipal("client")
Jungho Ahn19e84b22015-05-18 13:22:27 -070057 pclient2 := testutil.NewPrincipal("client2")
Asim Shankar4a698282015-03-21 21:59:18 -070058 pserver := testutil.NewPrincipal("server")
Jiri Simsa5293dcb2014-05-10 09:56:38 -070059
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -070060 ln, ep, err := server.Listen(protocol, "127.0.0.1:0", pserver, pserver.BlessingStore().Default())
Jiri Simsa5293dcb2014-05-10 09:56:38 -070061 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 Sivakumar2ad4e102015-03-17 21:23:37 -070069 if clientVC, err = client.Dial(ep, pclient); err != nil {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070070 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 Ahn19e84b22015-05-18 13:22:27 -0700134 // 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 Simsa5293dcb2014-05-10 09:56:38 -0700138 t.Errorf("Should not be able to Dial after listener is closed")
139 }
140}
141
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800142func TestSimpleFlow(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800143 testSimpleFlow(t, "tcp")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800144}
145
146func TestSimpleFlowWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800147 testSimpleFlow(t, "ws")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800148}
149
Cosmos Nicolaou9388ae42014-11-10 10:57:15 -0800150func 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 Rosencrantzbf0d9d92015-04-08 12:43:14 -0700157 _, err := client.Dial(ep, testutil.NewPrincipal("client"), DialTimeout(time.Second))
Cosmos Nicolaou9388ae42014-11-10 10:57:15 -0800158 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 Nicolaouae8dd212014-12-13 23:43:08 -0800171func testAuthenticatedByDefault(t *testing.T, protocol string) {
Asim Shankar6b0510a2014-10-01 12:05:06 -0700172 var (
173 server = InternalNew(naming.FixedRoutingID(0x55555555))
174 client = InternalNew(naming.FixedRoutingID(0xcccccccc))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700175
Asim Shankar4a698282015-03-21 21:59:18 -0700176 clientPrincipal = testutil.NewPrincipal("client")
177 serverPrincipal = testutil.NewPrincipal("server")
Suharsh Sivakumar59c423c2015-03-11 14:06:03 -0700178 clientKey = clientPrincipal.PublicKey()
179 serverBlessings = serverPrincipal.BlessingStore().Default()
Asim Shankar6b0510a2014-10-01 12:05:06 -0700180 )
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700181 ln, ep, err := server.Listen(protocol, "127.0.0.1:0", serverPrincipal, serverPrincipal.BlessingStore().Default())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700182 if err != nil {
183 t.Fatal(err)
184 }
Asim Shankar7171a252015-03-07 14:41:40 -0800185 // 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 Simsa5293dcb2014-05-10 09:56:38 -0700189
190 errs := make(chan error)
191
Ankur50a5f392015-02-27 18:46:30 -0800192 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 Simsa5293dcb2014-05-10 09:56:38 -0700203 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()
Ankur50a5f392015-02-27 18:46:30 -0800215 testAuth("server", flow, serverBlessings, clientKey)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700216 }()
217
218 go func() {
Suharsh Sivakumar2ad4e102015-03-17 21:23:37 -0700219 vc, err := client.Dial(ep, clientPrincipal)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700220 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()
Ankur50a5f392015-02-27 18:46:30 -0800230 testAuth("client", flow, serverBlessings, clientKey)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700231 }()
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 Jayaramandbae76b2014-11-17 12:51:29 -0800241func TestAuthenticatedByDefault(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800242 testAuthenticatedByDefault(t, "tcp")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800243}
244
245func TestAuthenticatedByDefaultWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800246 testAuthenticatedByDefault(t, "ws")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800247}
248
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700249func numListeners(m stream.Manager) int { return len(m.(*manager).listeners) }
250func debugString(m stream.Manager) string { return m.(*manager).DebugString() }
251func numVIFs(m stream.Manager) int { return len(m.(*manager).vifs.List()) }
252
253func TestListenEndpoints(t *testing.T) {
254 server := InternalNew(naming.FixedRoutingID(0xcafe))
Asim Shankar4a698282015-03-21 21:59:18 -0700255 principal := testutil.NewPrincipal("test")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700256 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 Sadovsky5181bdb2014-08-13 10:29:11 -0700259 // 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 Simsa5293dcb2014-05-10 09:56:38 -0700261 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
283func acceptLoop(ln stream.Listener) {
284 for {
285 f, err := ln.Accept()
286 if err != nil {
287 return
288 }
289 f.Close()
290 }
291}
292
293func TestCloseListener(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800294 testCloseListener(t, "tcp")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700295}
296
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800297func TestCloseListenerWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800298 testCloseListener(t, "ws")
299}
300
301func testCloseListener(t *testing.T, protocol string) {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800302 server := InternalNew(naming.FixedRoutingID(0x5e97e9))
Asim Shankar4a698282015-03-21 21:59:18 -0700303 pclient := testutil.NewPrincipal("client")
304 pserver := testutil.NewPrincipal("server")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700305 blessings := pserver.BlessingStore().Default()
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800306
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700307 ln, ep, err := server.Listen(protocol, "127.0.0.1:0", pserver, blessings)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800308 if err != nil {
309 t.Fatal(err)
310 }
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800311 // Server will just listen for flows and close them.
312 go acceptLoop(ln)
313 client := InternalNew(naming.FixedRoutingID(0xc1e41))
Suharsh Sivakumar2ad4e102015-03-17 21:23:37 -0700314 if _, err = client.Dial(ep, pclient); err != nil {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800315 t.Fatal(err)
316 }
317 ln.Close()
318 client = InternalNew(naming.FixedRoutingID(0xc1e42))
Suharsh Sivakumar2ad4e102015-03-17 21:23:37 -0700319 if _, err := client.Dial(ep, pclient); err == nil {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800320 t.Errorf("client.Dial(%q) should have failed", ep)
321 }
322}
323
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700324func TestShutdown(t *testing.T) {
325 server := InternalNew(naming.FixedRoutingID(0x5e97e9))
Asim Shankar4a698282015-03-21 21:59:18 -0700326 principal := testutil.NewPrincipal("test")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700327 blessings := principal.BlessingStore().Default()
328 ln, _, err := server.Listen("tcp", "127.0.0.1:0", principal, blessings)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700329 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 Sivakumare5e5dcc2015-03-18 14:29:31 -0700338 if _, _, err := server.Listen("tcp", "127.0.0.1:0", principal, blessings); err == nil {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700339 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
346func TestShutdownEndpoint(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800347 testShutdownEndpoint(t, "tcp")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700348}
349
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800350func TestShutdownEndpointWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800351 testShutdownEndpoint(t, "ws")
352}
353
354func testShutdownEndpoint(t *testing.T, protocol string) {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800355 server := InternalNew(naming.FixedRoutingID(0x55555555))
356 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700357 principal := testutil.NewPrincipal("test")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800358
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700359 ln, ep, err := server.Listen(protocol, "127.0.0.1:0", principal, principal.BlessingStore().Default())
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800360 if err != nil {
361 t.Fatal(err)
362 }
363
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800364 // Server will just listen for flows and close them.
365 go acceptLoop(ln)
366
Asim Shankar4a698282015-03-21 21:59:18 -0700367 vc, err := client.Dial(ep, testutil.NewPrincipal("client"))
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800368 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 Ahn6ab655f2015-04-14 18:27:09 -0700380func TestStartTimeout(t *testing.T) {
381 const (
382 startTime = 5 * time.Millisecond
Jungho Ahn6ab655f2015-04-14 18:27:09 -0700383 )
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 Ahncc9d5722015-04-22 10:13:04 -0700416 for range time.Tick(startTime) {
417 if numVIFs(server) == 0 {
418 break
Jungho Ahn6ab655f2015-04-14 18:27:09 -0700419 }
420 }
Jungho Ahn6ab655f2015-04-14 18:27:09 -0700421}
422
Jungho Ahncd175b82015-03-27 14:29:40 -0700423func 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 Ahn19e84b22015-05-18 13:22:27 -0700453 errch := make(chan error)
Jungho Ahncd175b82015-03-27 14:29:40 -0700454 go func() {
455 for {
456 _, err := ln.Accept()
Jungho Ahn19e84b22015-05-18 13:22:27 -0700457 errch <- err
Jungho Ahncd175b82015-03-27 14:29:40 -0700458 }
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 Ahn19e84b22015-05-18 13:22:27 -0700469 // 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 Ahncd175b82015-03-27 14:29:40 -0700473
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 Ahncc9d5722015-04-22 10:13:04 -0700489 for range time.Tick(idleTime) {
490 if numVIFs(client) == 0 && numVIFs(server) == 0 {
491 break
Jungho Ahncd175b82015-03-27 14:29:40 -0700492 }
493 }
Jungho Ahncd175b82015-03-27 14:29:40 -0700494}
495
496func TestIdleTimeout(t *testing.T) { testIdleTimeout(t, false) }
497func TestIdleTimeoutServer(t *testing.T) { testIdleTimeout(t, true) }
498
Andres Erbsenffa45742014-08-13 10:13:11 -0700499/* TLS + resumption + channel bindings is broken: <https://secure-resumption.com/#channelbindings>.
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700500func TestSessionTicketCache(t *testing.T) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700501 server := InternalNew(naming.FixedRoutingID(0x55555555))
Asim Shankar4a698282015-03-21 21:59:18 -0700502 _, ep, err := server.Listen("tcp", "127.0.0.1:0", testutil.NewPrincipal("server"))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700503 if err != nil {
504 t.Fatal(err)
505 }
506
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700507 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700508 if _, err = client.Dial(ep, testutil.NewPrincipal("TestSessionTicketCacheClient")); err != nil {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700509 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 Erbsenffa45742014-08-13 10:13:11 -0700516*/
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700517
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800518func testMultipleVCs(t *testing.T, protocol string) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700519 server := InternalNew(naming.FixedRoutingID(0x55555555))
520 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700521 principal := testutil.NewPrincipal("test")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700522
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 Sivakumare5e5dcc2015-03-18 14:29:31 -0700528 ln, ep, err := server.Listen(protocol, "127.0.0.1:0", principal, principal.BlessingStore().Default())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700529 if err != nil {
530 t.Fatal(err)
531 }
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800532
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700533 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 Shankar4a698282015-03-21 21:59:18 -0700565 vcs[i], err = client.Dial(ep, testutil.NewPrincipal("client"))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700566 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 Jayaramandbae76b2014-11-17 12:51:29 -0800599func TestMultipleVCs(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800600 testMultipleVCs(t, "tcp")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800601}
602
603func TestMultipleVCsWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800604 testMultipleVCs(t, "ws")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800605}
606
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700607func TestAddressResolution(t *testing.T) {
608 server := InternalNew(naming.FixedRoutingID(0x55555555))
609 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700610 principal := testutil.NewPrincipal("test")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700611
Adam Sadovsky5181bdb2014-08-13 10:29:11 -0700612 // 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 Simsa5293dcb2014-05-10 09:56:38 -0700615 // As of April 2014, https://developers.google.com/compute/docs/networking
616 // said that IPv6 is not yet supported.
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700617 ln, ep, err := server.Listen("tcp4", "127.0.0.1:0", principal, principal.BlessingStore().Default())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700618 if err != nil {
619 t.Fatal(err)
620 }
621 go acceptLoop(ln)
622
Adam Sadovsky5181bdb2014-08-13 10:29:11 -0700623 // 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 Sivakumar7e93ce52015-05-07 17:46:13 -0700626 // create a endpoint with "localhost", which will result in an endpoint that
627 // doesn't contain 127.0.0.1.
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700628 _, port, _ := net.SplitHostPort(ep.Addr().String())
Matt Rosencrantzc16339c2015-04-23 10:47:06 -0700629 nep := &inaming.Endpoint{
630 Protocol: ep.Addr().Network(),
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700631 Address: net.JoinHostPort("localhost", port),
Matt Rosencrantzc16339c2015-04-23 10:47:06 -0700632 RID: ep.RoutingID(),
633 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700634
635 // Dial multiple VCs
636 for i := 0; i < 2; i++ {
Asim Shankar4a698282015-03-21 21:59:18 -0700637 if _, err = client.Dial(nep, testutil.NewPrincipal("client")); err != nil {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700638 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
651func TestServerRestartDuringClientLifetime(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800652 testServerRestartDuringClientLifetime(t, "tcp")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700653}
654
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800655func TestServerRestartDuringClientLifetimeWS(t *testing.T) {
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800656 testServerRestartDuringClientLifetime(t, "ws")
657}
658
659func testServerRestartDuringClientLifetime(t *testing.T, protocol string) {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800660 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700661 pclient := testutil.NewPrincipal("client")
Jungho Ahn19e84b22015-05-18 13:22:27 -0700662 pclient2 := testutil.NewPrincipal("client2")
Cosmos Nicolaou9e909842015-03-17 11:58:59 -0700663 sh, err := modules.NewShell(nil, nil, testing.Verbose(), t)
Cosmos Nicolaou344cc4a2014-11-26 15:38:43 -0800664 if err != nil {
665 t.Fatalf("unexpected error: %s", err)
666 }
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800667 defer sh.Cleanup(nil, nil)
Todd Wang95873902015-05-22 14:21:30 -0700668 h, err := sh.Start(nil, runServer, protocol, "127.0.0.1:0")
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800669 if err != nil {
670 t.Fatalf("unexpected error: %s", err)
671 }
Asim Shankare89936f2015-04-22 17:37:43 -0700672 epstr := expect.NewSession(t, h.Stdout(), time.Minute).ExpectVar("ENDPOINT")
673 ep, err := inaming.NewEndpoint(epstr)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800674 if err != nil {
Asim Shankare89936f2015-04-22 17:37:43 -0700675 t.Fatalf("inaming.NewEndpoint(%q): %v", epstr, err)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800676 }
Suharsh Sivakumar2ad4e102015-03-17 21:23:37 -0700677 if _, err := client.Dial(ep, pclient); err != nil {
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800678 t.Fatal(err)
679 }
680 h.Shutdown(nil, os.Stderr)
681
Jungho Ahn19e84b22015-05-18 13:22:27 -0700682 // 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 Jayaramandbae76b2014-11-17 12:51:29 -0800686 t.Fatal("Expected client.Dial to fail since server is dead")
687 }
688
Todd Wang95873902015-05-22 14:21:30 -0700689 h, err = sh.Start(nil, runServer, protocol, ep.Addr().String())
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800690 if err != nil {
691 t.Fatalf("unexpected error: %s", err)
692 }
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800693 // Restarting the server, listening on the same address as before
Asim Shankare89936f2015-04-22 17:37:43 -0700694 ep2, err := inaming.NewEndpoint(expect.NewSession(t, h.Stdout(), time.Minute).ExpectVar("ENDPOINT"))
695 if err != nil {
696 t.Fatal(err)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800697 }
Asim Shankare89936f2015-04-22 17:37:43 -0700698 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 Jayaramandbae76b2014-11-17 12:51:29 -0800702 t.Fatal(err)
703 }
704}
705
Todd Wang95873902015-05-22 14:21:30 -0700706var runServer = modules.Register(runServerFunc, "runServer")
707
708func runServerFunc(env *modules.Env, args ...string) error {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700709 server := InternalNew(naming.FixedRoutingID(0x55555555))
Asim Shankar4a698282015-03-21 21:59:18 -0700710 principal := testutil.NewPrincipal("test")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700711 _, ep, err := server.Listen(args[0], args[1], principal, principal.BlessingStore().Default())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700712 if err != nil {
Todd Wang95873902015-05-22 14:21:30 -0700713 fmt.Fprintln(env.Stderr, err)
Cosmos Nicolaou920db002014-10-23 16:57:32 -0700714 return err
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700715 }
Todd Wang95873902015-05-22 14:21:30 -0700716 fmt.Fprintf(env.Stdout, "ENDPOINT=%v\n", ep)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700717 // Live forever (till the process is explicitly killed)
Todd Wang95873902015-05-22 14:21:30 -0700718 modules.WaitForEOF(env.Stdin)
Cosmos Nicolaou920db002014-10-23 16:57:32 -0700719 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700720}
721
Todd Wang95873902015-05-22 14:21:30 -0700722var runRLimitedServer = modules.Register(func(env *modules.Env, args ...string) error {
Asim Shankare89936f2015-04-22 17:37:43 -0700723 var rlimit syscall.Rlimit
724 if err := syscall.Getrlimit(syscall.RLIMIT_NOFILE, &rlimit); err != nil {
Todd Wang95873902015-05-22 14:21:30 -0700725 fmt.Fprintln(env.Stderr, err)
Asim Shankare89936f2015-04-22 17:37:43 -0700726 return err
727 }
728 rlimit.Cur = 9
729 if err := syscall.Setrlimit(syscall.RLIMIT_NOFILE, &rlimit); err != nil {
Todd Wang95873902015-05-22 14:21:30 -0700730 fmt.Fprintln(env.Stderr, err)
Asim Shankare89936f2015-04-22 17:37:43 -0700731 return err
732 }
Todd Wang95873902015-05-22 14:21:30 -0700733 fmt.Fprintf(env.Stdout, "RLIMIT_NOFILE=%d\n", rlimit.Cur)
734 return runServerFunc(env, args...)
735}, "runRLimitedServer")
Asim Shankare89936f2015-04-22 17:37:43 -0700736
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700737func 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
752func 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 Nicolaou5129fcb2014-08-22 11:52:25 -0700760
761func TestRegistration(t *testing.T) {
762 server := InternalNew(naming.FixedRoutingID(0x55555555))
763 client := InternalNew(naming.FixedRoutingID(0xcccccccc))
Asim Shankar4a698282015-03-21 21:59:18 -0700764 principal := testutil.NewPrincipal("server")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700765 blessings := principal.BlessingStore().Default()
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700766
Cosmos Nicolaou87c0a552014-12-02 23:05:49 -0800767 dialer := func(_, _ string, _ time.Duration) (net.Conn, error) {
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700768 return nil, fmt.Errorf("tn.Dial")
769 }
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700770 resolver := func(_, _ string) (string, string, error) {
771 return "", "", fmt.Errorf("tn.Resolve")
772 }
Cosmos Nicolaou87c0a552014-12-02 23:05:49 -0800773 listener := func(_, _ string) (net.Listener, error) {
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700774 return nil, fmt.Errorf("tn.Listen")
775 }
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700776 rpc.RegisterProtocol("tn", dialer, resolver, listener)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700777
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700778 _, _, err := server.Listen("tnx", "127.0.0.1:0", principal, blessings)
Cosmos Nicolaou07a736d2015-04-06 11:55:26 -0700779 if err == nil || !strings.Contains(err.Error(), "unknown network: tnx") {
780 t.Fatalf("expected error is missing (%v)", err)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700781 }
782
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700783 _, _, err = server.Listen("tn", "127.0.0.1:0", principal, blessings)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700784 if err == nil || !strings.Contains(err.Error(), "tn.Listen") {
Cosmos Nicolaou07a736d2015-04-06 11:55:26 -0700785 t.Fatalf("expected error is missing (%v)", err)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700786 }
787
788 // Need a functional listener to test Dial.
Cosmos Nicolaou87c0a552014-12-02 23:05:49 -0800789 listener = func(_, addr string) (net.Listener, error) {
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700790 return net.Listen("tcp", addr)
791 }
792
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700793 if got, want := rpc.RegisterProtocol("tn", dialer, resolver, listener), true; got != want {
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700794 t.Errorf("got %t, want %t", got, want)
795 }
796
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700797 _, ep, err := server.Listen("tn", "127.0.0.1:0", principal, blessings)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700798 if err != nil {
799 t.Errorf("unexpected error %s", err)
800 }
801
Asim Shankar4a698282015-03-21 21:59:18 -0700802 _, err = client.Dial(ep, testutil.NewPrincipal("client"))
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700803 if err == nil || !strings.Contains(err.Error(), "tn.Resolve") {
804 t.Fatalf("expected error is missing (%v)", err)
Cosmos Nicolaou5129fcb2014-08-22 11:52:25 -0700805 }
806}
Asim Shankar7171a252015-03-07 14:41:40 -0800807
808func TestBlessingNamesInEndpoint(t *testing.T) {
809 var (
Asim Shankar4a698282015-03-21 21:59:18 -0700810 p = testutil.NewPrincipal("default")
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700811 b, _ = p.BlessSelf("dev.v.io/users/foo@bar.com/devices/desktop/app/myapp")
Asim Shankar7171a252015-03-07 14:41:40 -0800812
813 server = InternalNew(naming.FixedRoutingID(0x1))
814
815 tests = []struct {
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700816 principal security.Principal
817 blessings security.Blessings
818 blessingNames []string
819 err bool
Asim Shankar7171a252015-03-07 14:41:40 -0800820 }{
821 {
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700822 // 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 Shankar7171a252015-03-07 14:41:40 -0800826 },
827 {
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700828 // It is an error to provide a principal without providing blessings.
Suharsh Sivakumar59c423c2015-03-11 14:06:03 -0700829 principal: p,
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700830 blessings: security.Blessings{},
Suharsh Sivakumar59c423c2015-03-11 14:06:03 -0700831 err: true,
Asim Shankar7171a252015-03-07 14:41:40 -0800832 },
833 {
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700834 // It is an error to provide inconsistent blessings and principal
Asim Shankar4a698282015-03-21 21:59:18 -0700835 principal: testutil.NewPrincipal("random"),
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700836 blessings: b,
Suharsh Sivakumar59c423c2015-03-11 14:06:03 -0700837 err: true,
Asim Shankar7171a252015-03-07 14:41:40 -0800838 },
839 }
840 )
841 // p must recognize its own blessings!
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700842 p.AddToRoots(b)
Asim Shankar7171a252015-03-07 14:41:40 -0800843 for idx, test := range tests {
Suharsh Sivakumare5e5dcc2015-03-18 14:29:31 -0700844 ln, ep, err := server.Listen("tcp", "127.0.0.1:0", test.principal, test.blessings)
Asim Shankar7171a252015-03-07 14:41:40 -0800845 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 Sivakumare5e5dcc2015-03-18 14:29:31 -0700852 got, want := ep.BlessingNames(), test.blessingNames
Suharsh Sivakumarad1d4192015-03-09 16:48:10 -0700853 sort.Strings(got)
854 sort.Strings(want)
855 if !reflect.DeepEqual(got, want) {
Asim Shankar7171a252015-03-07 14:41:40 -0800856 t.Errorf("test #%d: Got %v, want %v", idx, got, want)
857 }
858 }
859}
Asim Shankare89936f2015-04-22 17:37:43 -0700860
861func 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 Wang95873902015-05-22 14:21:30 -0700867 h, err := sh.Start(nil, runRLimitedServer, "--logtostderr=true", "tcp", "127.0.0.1:0")
Asim Shankare89936f2015-04-22 17:37:43 -0700868 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 Nicolaouc818b802015-06-05 15:52:45 -0700920 t.Logf("%s", stderr.String())
Asim Shankare89936f2015-04-22 17:37:43 -0700921 t.Fatal(err)
922 }
Matt Rosencrantza40f12a2015-04-24 11:07:55 -0700923 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 Shankare89936f2015-04-22 17:37:43 -0700925 }
926 t.Logf("Server FD limit:%d", nfiles)
927 t.Logf("Client connection attempts: %d", nattempts)
928}
Suharsh Sivakumar859ea0f2015-04-29 23:51:39 -0700929
930func 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 Sivakumar859ea0f2015-04-29 23:51:39 -0700947 nep := &inaming.Endpoint{
948 Protocol: ep.Addr().Network(),
Suharsh Sivakumar7e93ce52015-05-07 17:46:13 -0700949 Address: ep.Addr().String(),
Suharsh Sivakumar859ea0f2015-04-29 23:51:39 -0700950 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 Ahn19e84b22015-05-18 13:22:27 -0700957 _, err := client.Dial(nep, testutil.NewPrincipal("client"))
Suharsh Sivakumar859ea0f2015-04-29 23:51:39 -0700958 errCh <- err
959 }()
960 }
961 for i := 0; i < 10; i++ {
Jungho Ahn19e84b22015-05-18 13:22:27 -0700962 if err := <-errCh; err != nil {
Suharsh Sivakumar859ea0f2015-04-29 23:51:39 -0700963 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}