blob: 712e4afbc5e2dc967ba83edb449705e2051be513 [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
Cosmos Nicolaou4e029972014-06-13 14:53:08 -07005package namespace_test
Jiri Simsa5293dcb2014-05-10 09:56:38 -07006
7import (
Asim Shankar263c73b2015-03-19 18:31:26 -07008 "fmt"
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -07009 "runtime"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070010 "runtime/debug"
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -070011 "sync"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070012 "testing"
13 "time"
14
Jiri Simsa6ac95222015-02-23 16:11:49 -080015 "v.io/v23"
16 "v.io/v23/context"
Todd Wang5082a552015-04-02 10:56:11 -070017 "v.io/v23/namespace"
Jiri Simsa6ac95222015-02-23 16:11:49 -080018 "v.io/v23/naming"
19 "v.io/v23/options"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070020 "v.io/v23/rpc"
Jiri Simsa6ac95222015-02-23 16:11:49 -080021 "v.io/v23/security"
22 "v.io/v23/verror"
Jiri Simsa337af232015-02-27 14:36:46 -080023 "v.io/x/lib/vlog"
Cosmos Nicolaouf889c732014-10-16 20:46:54 -070024
Jiri Simsaffceefa2015-02-28 11:03:34 -080025 _ "v.io/x/ref/profiles"
Todd Wang5082a552015-04-02 10:56:11 -070026 inamespace "v.io/x/ref/profiles/internal/naming/namespace"
Todd Wang5987a942015-04-06 11:06:17 -070027 "v.io/x/ref/services/mounttable/mounttablelib"
Todd Wang5082a552015-04-02 10:56:11 -070028 "v.io/x/ref/test"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070029 "v.io/x/ref/test/testutil"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070030)
31
Suharsh Sivakumard19c95d2015-02-19 14:44:50 -080032//go:generate v23 test generate
Asim Shankarc920db32014-10-16 19:18:21 -070033
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080034func createContexts(t *testing.T) (sc, c *context.T, cleanup func()) {
Cosmos Nicolaoua18a1eb2015-03-12 13:15:01 -070035 ctx, shutdown := test.InitForTest()
Asim Shankarb547ea92015-02-17 18:49:45 -080036 var (
37 err error
Asim Shankar4a698282015-03-21 21:59:18 -070038 psc = testutil.NewPrincipal("sc")
39 pc = testutil.NewPrincipal("c")
Asim Shankarb547ea92015-02-17 18:49:45 -080040 )
41 // Setup the principals so that they recognize each other.
42 if err := psc.AddToRoots(pc.BlessingStore().Default()); err != nil {
Ankurfb9255a2015-01-21 15:29:38 -080043 t.Fatal(err)
44 }
Asim Shankarb547ea92015-02-17 18:49:45 -080045 if err := pc.AddToRoots(psc.BlessingStore().Default()); err != nil {
46 t.Fatal(err)
47 }
Jiri Simsa6ac95222015-02-23 16:11:49 -080048 if sc, err = v23.SetPrincipal(ctx, psc); err != nil {
Asim Shankarb547ea92015-02-17 18:49:45 -080049 t.Fatal(err)
50 }
Jiri Simsa6ac95222015-02-23 16:11:49 -080051 if c, err = v23.SetPrincipal(ctx, pc); err != nil {
Ankurfb9255a2015-01-21 15:29:38 -080052 t.Fatal(err)
53 }
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080054 return sc, c, shutdown
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -080055}
56
Jiri Simsa5293dcb2014-05-10 09:56:38 -070057func boom(t *testing.T, f string, v ...interface{}) {
58 t.Logf(f, v...)
59 t.Fatal(string(debug.Stack()))
60}
61
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -070062// N squared but who cares, this is a little test.
63// Ignores dups.
64func contains(container, contained []string) bool {
65L:
66 for _, d := range contained {
67 for _, r := range container {
68 if r == d {
69 continue L
70 }
71 }
72 return false
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -070073 }
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -070074 return true
75}
76
77func compare(t *testing.T, caller, name string, got, want []string) {
78 // Compare ignoring dups.
79 if !contains(got, want) || !contains(want, got) {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -070080 boom(t, "%s: %q: got %v, want %v", caller, name, got, want)
81 }
82}
83
Todd Wang5082a552015-04-02 10:56:11 -070084func doGlob(t *testing.T, ctx *context.T, ns namespace.T, pattern string, limit int) []string {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070085 var replies []string
Matt Rosencrantz6edab562015-01-12 11:07:55 -080086 rc, err := ns.Glob(ctx, pattern)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070087 if err != nil {
88 boom(t, "Glob(%s): %s", pattern, err)
89 }
90 for s := range rc {
David Why Use Two When One Will Do Presotto03c34d62015-02-10 01:38:58 -080091 switch v := s.(type) {
James Ringc971dca2015-04-09 09:51:20 -070092 case *naming.GlobReplyEntry:
93 replies = append(replies, v.Value.Name)
David Why Use Two When One Will Do Presotto03c34d62015-02-10 01:38:58 -080094 if limit > 0 && len(replies) > limit {
95 boom(t, "Glob returns too many results, perhaps not limiting recursion")
96 }
David Why Use Two When One Will Do Presotto2db32f92014-07-18 12:20:35 -070097 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -070098 }
99 return replies
100}
101
Robin Thellend434e39f2014-08-27 14:46:27 -0700102type testServer struct {
103 suffix string
104}
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700105
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700106func (testServer) KnockKnock(call rpc.ServerCall) (string, error) {
Todd Wange77f9952015-02-18 13:20:50 -0800107 return "Who's there?", nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700108}
109
Robin Thellend39ac3232014-12-02 09:50:41 -0800110// testServer has the following namespace:
Robin Thellend434e39f2014-08-27 14:46:27 -0700111// "" -> {level1} -> {level2}
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700112func (t *testServer) GlobChildren__(rpc.ServerCall) (<-chan string, error) {
Robin Thellend39ac3232014-12-02 09:50:41 -0800113 ch := make(chan string, 1)
114 switch t.suffix {
115 case "":
116 ch <- "level1"
117 case "level1":
118 ch <- "level2"
119 default:
120 return nil, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700121 }
Robin Thellend39ac3232014-12-02 09:50:41 -0800122 close(ch)
123 return ch, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700124}
125
Asim Shankar98d65962014-10-24 16:55:30 -0700126type allowEveryoneAuthorizer struct{}
127
Matt Rosencrantz250558f2015-03-17 11:37:31 -0700128func (allowEveryoneAuthorizer) Authorize(*context.T) error { return nil }
Asim Shankar98d65962014-10-24 16:55:30 -0700129
Robin Thellend434e39f2014-08-27 14:46:27 -0700130type dispatcher struct{}
131
Robin Thellenda02fe8f2014-11-19 09:58:29 -0800132func (d *dispatcher) Lookup(suffix string) (interface{}, security.Authorizer, error) {
Cosmos Nicolaou710daa22014-11-11 19:39:18 -0800133 return &testServer{suffix}, allowEveryoneAuthorizer{}, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700134}
135
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800136func knockKnock(t *testing.T, ctx *context.T, name string) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800137 client := v23.GetClient(ctx)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700138 var result string
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -0700139 if err := client.Call(ctx, name, "KnockKnock", nil, []interface{}{&result}); err != nil {
140 boom(t, "Call failed: %s", err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700141 }
142 if result != "Who's there?" {
143 boom(t, "Wrong result: %v", result)
144 }
145}
146
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700147func doResolveTest(t *testing.T, fname string, f func(*context.T, string, ...naming.NamespaceOpt) (*naming.MountEntry, error), ctx *context.T, name string, want []string, opts ...naming.NamespaceOpt) {
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800148 me, err := f(ctx, name, opts...)
Ryan Brownbc2c87c2014-11-17 18:55:25 +0000149 if err != nil {
Ryan Brownfc290142014-11-24 15:47:24 -0800150 boom(t, "Failed to %s %s: %s", fname, name, err)
Ryan Brownbc2c87c2014-11-17 18:55:25 +0000151 }
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800152 compare(t, fname, name, me.Names(), want)
Ryan Brownfc290142014-11-24 15:47:24 -0800153}
154
Todd Wang5082a552015-04-02 10:56:11 -0700155func testResolveToMountTable(t *testing.T, ctx *context.T, ns namespace.T, name string, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800156 doResolveTest(t, "ResolveToMountTable", ns.ResolveToMountTable, ctx, name, want)
Ryan Brownfc290142014-11-24 15:47:24 -0800157}
158
Todd Wang5082a552015-04-02 10:56:11 -0700159func testResolveToMountTableWithPattern(t *testing.T, ctx *context.T, ns namespace.T, name string, pattern naming.NamespaceOpt, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800160 doResolveTest(t, "ResolveToMountTable", ns.ResolveToMountTable, ctx, name, want, pattern)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700161}
162
Todd Wang5082a552015-04-02 10:56:11 -0700163func testResolve(t *testing.T, ctx *context.T, ns namespace.T, name string, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800164 doResolveTest(t, "Resolve", ns.Resolve, ctx, name, want)
Ryan Brownfc290142014-11-24 15:47:24 -0800165}
166
Todd Wang5082a552015-04-02 10:56:11 -0700167func testResolveWithPattern(t *testing.T, ctx *context.T, ns namespace.T, name string, pattern naming.NamespaceOpt, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800168 doResolveTest(t, "Resolve", ns.Resolve, ctx, name, want, pattern)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700169}
170
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700171type serverEntry struct {
172 mountPoint string
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700173 server rpc.Server
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700174 endpoint naming.Endpoint
175 name string
176}
177
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700178func runServer(t *testing.T, ctx *context.T, disp rpc.Dispatcher, mountPoint string) *serverEntry {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800179 return run(t, ctx, disp, mountPoint, false)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700180}
181
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800182func runMT(t *testing.T, ctx *context.T, mountPoint string) *serverEntry {
Todd Wang5987a942015-04-06 11:06:17 -0700183 mtd, err := mounttablelib.NewMountTableDispatcher("")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700184 if err != nil {
Bogdan Capritae96cd042015-02-03 17:32:57 -0800185 boom(t, "NewMountTableDispatcher returned error: %v", err)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700186 }
Bogdan Capritae96cd042015-02-03 17:32:57 -0800187 return run(t, ctx, mtd, mountPoint, true)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700188}
189
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700190func run(t *testing.T, ctx *context.T, disp rpc.Dispatcher, mountPoint string, mt bool) *serverEntry {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800191 s, err := v23.NewServer(ctx, options.ServesMountTable(mt))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700192 if err != nil {
193 boom(t, "r.NewServer: %s", err)
194 }
195 // Add a mount table server.
196 // Start serving on a loopback address.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800197 eps, err := s.Listen(v23.GetListenSpec(ctx))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700198 if err != nil {
199 boom(t, "Failed to Listen: %s", err)
200 }
Cosmos Nicolaou92dba582014-11-05 17:24:10 -0800201 if err := s.ServeDispatcher(mountPoint, disp); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700202 boom(t, "Failed to serve mount table at %s: %s", mountPoint, err)
203 }
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800204 return &serverEntry{mountPoint: mountPoint, server: s, endpoint: eps[0], name: eps[0].Name()}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700205}
206
207const (
208 mt1MP = "mt1"
209 mt2MP = "mt2"
210 mt3MP = "mt3"
211 mt4MP = "mt4"
212 mt5MP = "mt5"
213 j1MP = "joke1"
214 j2MP = "joke2"
215 j3MP = "joke3"
216
217 ttl = 100 * time.Second
218)
219
220// runMountTables creates a root mountable with some mount tables mounted
221// in it: mt{1,2,3,4,5}
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800222func runMountTables(t *testing.T, ctx *context.T) (*serverEntry, map[string]*serverEntry) {
223 root := runMT(t, ctx, "")
Jiri Simsa6ac95222015-02-23 16:11:49 -0800224 v23.GetNamespace(ctx).SetRoots(root.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700225 t.Logf("mountTable %q -> %s", root.mountPoint, root.endpoint)
226
227 mps := make(map[string]*serverEntry)
228 for _, mp := range []string{mt1MP, mt2MP, mt3MP, mt4MP, mt5MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800229 m := runMT(t, ctx, mp)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700230 t.Logf("mountTable %q -> %s", mp, m.endpoint)
231 mps[mp] = m
232 }
233 return root, mps
234}
235
Robin Thellend434e39f2014-08-27 14:46:27 -0700236// createNamespace creates a hierarchy of mounttables and servers
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700237// as follows:
238// /mt1, /mt2, /mt3, /mt4, /mt5, /joke1, /joke2, /joke3.
239// That is, mt1 is a mount table mounted in the root mount table,
240// joke1 is a server mounted in the root mount table.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800241func createNamespace(t *testing.T, ctx *context.T) (*serverEntry, map[string]*serverEntry, map[string]*serverEntry, func()) {
242 root, mts := runMountTables(t, ctx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700243 jokes := make(map[string]*serverEntry)
244 // Let's run some non-mount table services.
245 for _, j := range []string{j1MP, j2MP, j3MP} {
Robin Thellend434e39f2014-08-27 14:46:27 -0700246 disp := &dispatcher{}
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800247 jokes[j] = runServer(t, ctx, disp, j)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700248 }
249 return root, mts, jokes, func() {
250 for _, s := range jokes {
251 s.server.Stop()
252 }
253 for _, s := range mts {
254 s.server.Stop()
255 }
256 root.server.Stop()
257 }
258}
259
260// runNestedMountTables creates some nested mount tables in the hierarchy
261// created by createNamespace as follows:
262// /mt4/foo, /mt4/foo/bar and /mt4/baz where foo, bar and baz are mount tables.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800263func runNestedMountTables(t *testing.T, ctx *context.T, mts map[string]*serverEntry) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800264 ns := v23.GetNamespace(ctx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700265 // Set up some nested mounts and verify resolution.
266 for _, m := range []string{"mt4/foo", "mt4/foo/bar"} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800267 mts[m] = runMT(t, ctx, m)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700268 }
269
270 // Use a global name for a mount, rather than a relative one.
271 // We directly mount baz into the mt4/foo mount table.
272 globalMP := naming.JoinAddressName(mts["mt4/foo"].name, "baz")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800273 mts["baz"] = runMT(t, ctx, "baz")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700274 if err := ns.Mount(ctx, globalMP, mts["baz"].name, ttl); err != nil {
275 boom(t, "Failed to Mount %s: %s", globalMP, err)
276 }
277}
278
279// TestNamespaceCommon tests common use of the Namespace library
280// against a root mount table and some mount tables mounted on it.
281func TestNamespaceCommon(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800282 _, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800283 defer cleanup()
284
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800285 root, mts, jokes, stopper := createNamespace(t, c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700286 defer stopper()
Jiri Simsa6ac95222015-02-23 16:11:49 -0800287 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700288
289 // All of the initial mounts are served by the root mounttable
290 // and hence ResolveToMountTable should return the root mountable
291 // as the address portion of the terminal name for those mounttables.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800292 testResolveToMountTable(t, c, ns, "", root.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700293 for _, m := range []string{mt2MP, mt3MP, mt5MP} {
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800294 rootMT := naming.Join(root.name, m)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700295 // All of these mount tables are hosted by the root mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800296 testResolveToMountTable(t, c, ns, m, rootMT)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700297
298 // The server registered for each mount point is a mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800299 testResolve(t, c, ns, m, mts[m].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700300
301 // ResolveToMountTable will walk through to the sub MountTables
302 mtbar := naming.Join(m, "bar")
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800303 subMT := naming.Join(mts[m].name, "bar")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800304 testResolveToMountTable(t, c, ns, mtbar, subMT)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700305 }
306
307 for _, j := range []string{j1MP, j2MP, j3MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800308 testResolve(t, c, ns, j, jokes[j].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700309 }
310}
311
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800312// TestNamespaceDetails tests more detailed use of the Namespace library.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700313func TestNamespaceDetails(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800314 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800315 defer cleanup()
316
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800317 root, mts, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700318 defer stopper()
319
Jiri Simsa6ac95222015-02-23 16:11:49 -0800320 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700321 ns.SetRoots(root.name)
322
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800323 // /mt2 is not an endpoint. Thus, the example below will fail.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700324 mt3Server := mts[mt3MP].name
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800325 mt2a := "/mt2/a"
Todd Wang8fa38762015-03-25 14:04:59 -0700326 if err := ns.Mount(c, mt2a, mt3Server, ttl); verror.ErrorID(err) == naming.ErrNoSuchName.ID {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700327 boom(t, "Successfully mounted %s - expected an err %v, not %v", mt2a, naming.ErrNoSuchName, err)
328 }
329
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800330 // Mount using the relative name.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700331 // This means walk through mt2 if it already exists and mount within
332 // the lower level mount table, if the name doesn't exist we'll create
333 // a new name for it.
334 mt2a = "mt2/a"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800335 if err := ns.Mount(c, mt2a, mt3Server, ttl); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700336 boom(t, "Failed to Mount %s: %s", mt2a, err)
337 }
338
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800339 mt2mt := naming.Join(mts[mt2MP].name, "a")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700340 // The mt2/a is served by the mt2 mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800341 testResolveToMountTable(t, c, ns, mt2a, mt2mt)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700342 // The server for mt2a is mt3server from the second mount above.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800343 testResolve(t, c, ns, mt2a, mt3Server)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700344
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800345 // Add two more mounts. The // should be stripped off of the
346 // second.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700347 for _, mp := range []struct{ name, server string }{
348 {"mt2", mts[mt4MP].name},
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800349 {"mt2//", mts[mt5MP].name},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700350 } {
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700351 if err := ns.Mount(c, mp.name, mp.server, ttl, naming.ServesMountTable(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700352 boom(t, "Failed to Mount %s: %s", mp.name, err)
353 }
354 }
355
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800356 names := []string{naming.JoinAddressName(mts[mt4MP].name, "a"),
357 naming.JoinAddressName(mts[mt5MP].name, "a")}
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800358 names = append(names, naming.JoinAddressName(mts[mt2MP].name, "a"))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700359 // We now have 3 mount tables prepared to serve mt2/a
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800360 testResolveToMountTable(t, c, ns, "mt2/a", names...)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800361 names = []string{mts[mt4MP].name, mts[mt5MP].name}
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800362 names = append(names, mts[mt2MP].name)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800363 testResolve(t, c, ns, "mt2", names...)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700364}
365
366// TestNestedMounts tests some more deeply nested mounts
367func TestNestedMounts(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800368 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800369 defer cleanup()
370
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800371 root, mts, _, stopper := createNamespace(t, sc)
372 runNestedMountTables(t, sc, mts)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700373 defer stopper()
374
Jiri Simsa6ac95222015-02-23 16:11:49 -0800375 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700376 ns.SetRoots(root.name)
377
378 // Set up some nested mounts and verify resolution.
379 for _, m := range []string{"mt4/foo", "mt4/foo/bar"} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800380 testResolve(t, c, ns, m, mts[m].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700381 }
382
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800383 testResolveToMountTable(t, c, ns, "mt4/foo",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800384 naming.JoinAddressName(mts[mt4MP].name, "foo"))
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800385 testResolveToMountTable(t, c, ns, "mt4/foo/bar",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800386 naming.JoinAddressName(mts["mt4/foo"].name, "bar"))
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800387 testResolveToMountTable(t, c, ns, "mt4/foo/baz",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800388 naming.JoinAddressName(mts["mt4/foo"].name, "baz"))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700389}
390
391// TestServers tests invoking RPCs on simple servers
392func TestServers(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800393 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800394 defer cleanup()
395
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800396 root, mts, jokes, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700397 defer stopper()
Jiri Simsa6ac95222015-02-23 16:11:49 -0800398 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700399 ns.SetRoots(root.name)
400
Robin Thellend434e39f2014-08-27 14:46:27 -0700401 // Let's run some non-mount table services
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700402 for _, j := range []string{j1MP, j2MP, j3MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800403 testResolve(t, c, ns, j, jokes[j].name)
404 knockKnock(t, c, j)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700405 globalName := naming.JoinAddressName(mts["mt4"].name, j)
Robin Thellend434e39f2014-08-27 14:46:27 -0700406 disp := &dispatcher{}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700407 gj := "g_" + j
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800408 jokes[gj] = runServer(t, c, disp, globalName)
409 testResolve(t, c, ns, "mt4/"+j, jokes[gj].name)
410 knockKnock(t, c, "mt4/"+j)
411 testResolveToMountTable(t, c, ns, "mt4/"+j, globalName)
412 testResolveToMountTable(t, c, ns, "mt4/"+j+"/garbage", globalName+"/garbage")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700413 }
414}
415
416// TestGlob tests some glob patterns.
417func TestGlob(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800418 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800419 defer cleanup()
420
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800421 root, mts, _, stopper := createNamespace(t, sc)
422 runNestedMountTables(t, sc, mts)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700423 defer stopper()
Jiri Simsa6ac95222015-02-23 16:11:49 -0800424 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700425 ns.SetRoots(root.name)
426
427 tln := []string{"baz", "mt1", "mt2", "mt3", "mt4", "mt5", "joke1", "joke2", "joke3"}
428 barbaz := []string{"mt4/foo/bar", "mt4/foo/baz"}
Robin Thellend434e39f2014-08-27 14:46:27 -0700429 level12 := []string{"joke1/level1", "joke1/level1/level2", "joke2/level1", "joke2/level1/level2", "joke3/level1", "joke3/level1/level2"}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700430 foo := append([]string{"mt4/foo"}, barbaz...)
Robin Thellend434e39f2014-08-27 14:46:27 -0700431 foo = append(foo, level12...)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700432 // Try various globs.
433 globTests := []struct {
434 pattern string
435 expected []string
436 }{
437 {"*", tln},
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -0700438 {"x", []string{}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700439 {"m*", []string{"mt1", "mt2", "mt3", "mt4", "mt5"}},
440 {"mt[2,3]", []string{"mt2", "mt3"}},
441 {"*z", []string{"baz"}},
Robin Thellend434e39f2014-08-27 14:46:27 -0700442 {"joke1/*", []string{"joke1/level1"}},
443 {"j?ke1/level1/*", []string{"joke1/level1/level2"}},
444 {"joke1/level1/*", []string{"joke1/level1/level2"}},
445 {"joke1/level1/level2/...", []string{"joke1/level1/level2"}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700446 {"...", append(append(tln, foo...), "")},
447 {"*/...", append(tln, foo...)},
448 {"*/foo/*", barbaz},
449 {"*/*/*z", []string{"mt4/foo/baz"}},
450 {"*/f??/*z", []string{"mt4/foo/baz"}},
Robin Thellend434e39f2014-08-27 14:46:27 -0700451 {"mt4/foo/baz", []string{"mt4/foo/baz"}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700452 }
453 for _, test := range globTests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800454 out := doGlob(t, c, ns, test.pattern, 0)
Robin Thellend434e39f2014-08-27 14:46:27 -0700455 compare(t, "Glob", test.pattern, out, test.expected)
Robin Thellend5f838572014-07-10 13:28:56 -0700456 // Do the same with a full rooted name.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800457 out = doGlob(t, c, ns, naming.JoinAddressName(root.name, test.pattern), 0)
Robin Thellend5f838572014-07-10 13:28:56 -0700458 var expectedWithRoot []string
459 for _, s := range test.expected {
460 expectedWithRoot = append(expectedWithRoot, naming.JoinAddressName(root.name, s))
461 }
Robin Thellend434e39f2014-08-27 14:46:27 -0700462 compare(t, "Glob", test.pattern, out, expectedWithRoot)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700463 }
464}
465
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700466type GlobbableServer struct {
467 callCount int
468 mu sync.Mutex
469}
470
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700471func (g *GlobbableServer) Glob__(rpc.ServerCall, string) (<-chan naming.GlobReply, error) {
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700472 g.mu.Lock()
473 defer g.mu.Unlock()
474 g.callCount++
Robin Thellend39ac3232014-12-02 09:50:41 -0800475 return nil, nil
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700476}
477
478func (g *GlobbableServer) GetAndResetCount() int {
479 g.mu.Lock()
480 defer g.mu.Unlock()
481 cnt := g.callCount
482 g.callCount = 0
483
484 return cnt
485}
486
487// TestGlobEarlyStop tests that Glob doesn't query terminal servers with finished patterns.
488func TestGlobEarlyStop(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800489 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800490 defer cleanup()
491
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800492 root, mts, _, stopper := createNamespace(t, sc)
493 runNestedMountTables(t, sc, mts)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700494 defer stopper()
495
496 globServer := &GlobbableServer{}
497 name := naming.JoinAddressName(mts["mt4/foo/bar"].name, "glob")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800498 runningGlobServer := runServer(t, c, testutil.LeafDispatcher(globServer, nil), name)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700499 defer runningGlobServer.server.Stop()
500
Jiri Simsa6ac95222015-02-23 16:11:49 -0800501 ns := v23.GetNamespace(c)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700502 ns.SetRoots(root.name)
503
504 tests := []struct {
505 pattern string
506 expectedCalls int
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800507 expected []string
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700508 }{
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800509 {"mt4/foo/bar/glob", 0, []string{"mt4/foo/bar/glob"}},
510 {"mt4/foo/bar/glob/...", 1, []string{"mt4/foo/bar/glob"}},
511 {"mt4/foo/bar/glob/*", 1, nil},
512 {"mt4/foo/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
513 {"mt4/foo/bar/...", 1, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
514 {"mt4/foo/bar/*", 0, []string{"mt4/foo/bar/glob"}},
515 {"mt4/***/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
516 {"mt4/*/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700517 }
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800518 // Test allowing the tests to descend into leaves.
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700519 for _, test := range tests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800520 out := doGlob(t, c, ns, test.pattern, 0)
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800521 compare(t, "Glob", test.pattern, out, test.expected)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700522 if calls := globServer.GetAndResetCount(); calls != test.expectedCalls {
523 boom(t, "Wrong number of Glob calls to terminal server got: %d want: %d.", calls, test.expectedCalls)
524 }
525 }
526}
527
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700528func TestCycles(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800529 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800530 defer cleanup()
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700531
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800532 root, _, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700533 defer stopper()
Jiri Simsa6ac95222015-02-23 16:11:49 -0800534 ns := v23.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700535 ns.SetRoots(root.name)
536
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800537 c1 := runMT(t, c, "c1")
538 c2 := runMT(t, c, "c2")
539 c3 := runMT(t, c, "c3")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700540 defer c1.server.Stop()
541 defer c2.server.Stop()
542 defer c3.server.Stop()
543
544 m := "c1/c2"
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700545 if err := ns.Mount(c, m, c1.name, ttl, naming.ServesMountTable(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700546 boom(t, "Failed to Mount %s: %s", "c1/c2", err)
547 }
548
549 m = "c1/c2/c3"
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700550 if err := ns.Mount(c, m, c3.name, ttl, naming.ServesMountTable(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700551 boom(t, "Failed to Mount %s: %s", m, err)
552 }
553
554 m = "c1/c3/c4"
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700555 if err := ns.Mount(c, m, c1.name, ttl, naming.ServesMountTable(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700556 boom(t, "Failed to Mount %s: %s", m, err)
557 }
558
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800559 // Since c1 was mounted with the Serve call, it will have both the tcp and ws endpoints.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800560 testResolve(t, c, ns, "c1", c1.name)
561 testResolve(t, c, ns, "c1/c2", c1.name)
562 testResolve(t, c, ns, "c1/c3", c3.name)
563 testResolve(t, c, ns, "c1/c3/c4", c1.name)
564 testResolve(t, c, ns, "c1/c3/c4/c3/c4", c1.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700565 cycle := "c3/c4"
566 for i := 0; i < 40; i++ {
567 cycle += "/c3/c4"
568 }
Todd Wang8fa38762015-03-25 14:04:59 -0700569 if _, err := ns.Resolve(c, "c1/"+cycle); verror.ErrorID(err) != naming.ErrResolutionDepthExceeded.ID {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700570 boom(t, "Failed to detect cycle")
571 }
572
David Why Use Two When One Will Do Presotto3fc13f42015-03-30 09:56:26 -0700573 // Perform the glob with a response length limit and dup suppression. The dup supression
574 // should win.
575 r := doGlob(t, c, ns, "c1/...", 1000)
576 if len(r) != 6 {
577 t.Fatal("expected 6 replies, got %v", r)
578 }
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700579}
580
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700581// TestGoroutineLeaks tests for leaking goroutines - we have many:-(
582func TestGoroutineLeaks(t *testing.T) {
583 t.Skip()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800584 sc, _, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800585 defer cleanup()
586
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800587 _, _, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700588 defer func() {
589 vlog.Infof("%d goroutines:", runtime.NumGoroutine())
590 }()
591 defer stopper()
592 defer func() {
593 vlog.Infof("%d goroutines:", runtime.NumGoroutine())
594 }()
595 //panic("this will show up lots of goroutine+channel leaks!!!!")
596}
597
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700598func TestBadRoots(t *testing.T) {
Todd Wang5082a552015-04-02 10:56:11 -0700599 if _, err := inamespace.New(); err != nil {
Cosmos Nicolaou4e029972014-06-13 14:53:08 -0700600 t.Errorf("namespace.New should not have failed with no roots")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700601 }
Todd Wang5082a552015-04-02 10:56:11 -0700602 if _, err := inamespace.New("not a rooted name"); err == nil {
Cosmos Nicolaou4e029972014-06-13 14:53:08 -0700603 t.Errorf("namespace.New should have failed with an unrooted name")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700604 }
605}
Ryan Brownfc290142014-11-24 15:47:24 -0800606
607func bless(blesser, delegate security.Principal, extension string) {
608 b, err := blesser.Bless(delegate.PublicKey(), blesser.BlessingStore().Default(), extension, security.UnconstrainedUse())
609 if err != nil {
610 panic(err)
611 }
612 delegate.BlessingStore().SetDefault(b)
613}
614
Asim Shankar263c73b2015-03-19 18:31:26 -0700615func TestAuthorizationDuringResolve(t *testing.T) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800616 ctx, shutdown := v23.Init()
Asim Shankarb547ea92015-02-17 18:49:45 -0800617 defer shutdown()
618
619 var (
Asim Shankar4a698282015-03-21 21:59:18 -0700620 rootMtCtx, _ = v23.SetPrincipal(ctx, testutil.NewPrincipal()) // root mounttable
621 mtCtx, _ = v23.SetPrincipal(ctx, testutil.NewPrincipal()) // intermediate mounttable
622 serverCtx, _ = v23.SetPrincipal(ctx, testutil.NewPrincipal()) // end server
623 clientCtx, _ = v23.SetPrincipal(ctx, testutil.NewPrincipal()) // client process (doing Resolves).
624 idp = testutil.NewIDProvider("idp") // identity provider
Asim Shankar263c73b2015-03-19 18:31:26 -0700625 serverEndpoint = naming.FormatEndpoint("tcp", "127.0.0.1:14141")
Asim Shankarb547ea92015-02-17 18:49:45 -0800626
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700627 resolve = func(name string, opts ...naming.NamespaceOpt) (*naming.MountEntry, error) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800628 return v23.GetNamespace(clientCtx).Resolve(clientCtx, name, opts...)
Asim Shankarb547ea92015-02-17 18:49:45 -0800629 }
630
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700631 mount = func(name, server string, ttl time.Duration, opts ...naming.NamespaceOpt) error {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800632 return v23.GetNamespace(serverCtx).Mount(serverCtx, name, server, ttl, opts...)
Asim Shankarb547ea92015-02-17 18:49:45 -0800633 }
634 )
635 // Setup default blessings for the processes.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800636 idp.Bless(v23.GetPrincipal(rootMtCtx), "rootmt")
637 idp.Bless(v23.GetPrincipal(serverCtx), "server")
638 idp.Bless(v23.GetPrincipal(mtCtx), "childmt")
639 idp.Bless(v23.GetPrincipal(clientCtx), "client")
Asim Shankarb547ea92015-02-17 18:49:45 -0800640
641 // Setup the namespace root for all the "processes".
642 rootmt := runMT(t, rootMtCtx, "")
643 for _, ctx := range []*context.T{mtCtx, serverCtx, clientCtx} {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800644 v23.GetNamespace(ctx).SetRoots(rootmt.name)
Asim Shankarb547ea92015-02-17 18:49:45 -0800645 }
646 // Disable caching in the client so that any Mount calls by the server
647 // are noticed immediately.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800648 v23.GetNamespace(clientCtx).CacheCtl(naming.DisableCache(true))
Asim Shankarb547ea92015-02-17 18:49:45 -0800649
Asim Shankarb547ea92015-02-17 18:49:45 -0800650 // Intermediate mounttables should be authenticated.
Suharsh Sivakumar77357d72015-02-21 22:14:20 -0800651 mt := runMT(t, mtCtx, "mt")
652 // Mount a server on "mt".
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700653 if err := mount("mt/server", serverEndpoint, time.Minute, naming.ReplaceMount(true)); err != nil {
Asim Shankarb547ea92015-02-17 18:49:45 -0800654 t.Error(err)
Suharsh Sivakumar77357d72015-02-21 22:14:20 -0800655 }
Asim Shankar263c73b2015-03-19 18:31:26 -0700656 // The namespace root should be authenticated too
657 if e, err := resolve("mt/server"); err != nil {
658 t.Errorf("Got (%v, %v)", e, err)
659 } else {
660 // Host:Port and Endpoint versions of the other namespace root
661 // (which has different blessings)
662 hproot := fmt.Sprintf("(otherroot)@%v", rootmt.endpoint.Addr())
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700663 eproot := naming.FormatEndpoint(rootmt.endpoint.Addr().Network(), rootmt.endpoint.Addr().String(), rootmt.endpoint.RoutingID(), naming.BlessingOpt("otherroot"), naming.ServesMountTable(rootmt.endpoint.ServesMountTable()))
Asim Shankar263c73b2015-03-19 18:31:26 -0700664 for _, root := range []string{hproot, eproot} {
665 name := naming.JoinAddressName(root, "mt")
666 // Rooted name resolutions should fail authorization because of the "otherrot"
Todd Wang8fa38762015-03-25 14:04:59 -0700667 if e, err := resolve(name); verror.ErrorID(err) != verror.ErrNotTrusted.ID {
Asim Shankar263c73b2015-03-19 18:31:26 -0700668 t.Errorf("resolve(%q) returned (%v, errorid=%v %v), wanted errorid=%v", name, e, verror.ErrorID(err), err, verror.ErrNotTrusted.ID)
669 }
670 // But not fail if the skip-authorization option is provided
671 if e, err := resolve(name, options.SkipServerEndpointAuthorization{}); err != nil {
672 t.Errorf("resolve(%q): Got (%v, %v), expected resolution to succeed", name, e, err)
673 }
674 // The namespace root from the context should be authorized as well.
675 ctx, ns, _ := v23.SetNewNamespace(clientCtx, naming.JoinAddressName(root, ""))
Todd Wang8fa38762015-03-25 14:04:59 -0700676 if e, err := ns.Resolve(ctx, "mt/server"); verror.ErrorID(err) != verror.ErrNotTrusted.ID {
Asim Shankar263c73b2015-03-19 18:31:26 -0700677 t.Errorf("resolve with root=%q returned (%v, errorid=%v %v), wanted errorid=%v", root, e, verror.ErrorID(err), err, verror.ErrNotTrusted.ID)
678 }
679 if _, err := ns.Resolve(ctx, "mt/server", options.SkipServerEndpointAuthorization{}); err != nil {
680 t.Errorf("resolve with root=%q should have succeeded when authorization checks are skipped. Got %v", err)
681 }
682 }
683 }
684 // Imagine that the network address of "mt" has been taken over by an
685 // attacker. However, this attacker cannot mess with the mount entry
686 // for "mt". This would result in "mt" and its mount entry (in the
687 // global mounttable) having inconsistent blessings. Simulate this by
688 // explicitly changing the mount entry for "mt".
689 goodChildMTEndpoint := naming.FormatEndpoint(mt.endpoint.Addr().Network(), mt.endpoint.Addr().String(), naming.BlessingOpt("idp/goodchildmt"), mt.endpoint.RoutingID())
David Why Use Two When One Will Do Presotto38788d42015-03-31 17:13:54 -0700690 if err := v23.GetNamespace(mtCtx).Mount(mtCtx, "mt", goodChildMTEndpoint, time.Minute, naming.ServesMountTable(true), naming.ReplaceMount(true)); err != nil {
Asim Shankarb547ea92015-02-17 18:49:45 -0800691 t.Error(err)
Suharsh Sivakumar77357d72015-02-21 22:14:20 -0800692 }
693
Asim Shankar263c73b2015-03-19 18:31:26 -0700694 if e, err := resolve("mt/server", options.SkipServerEndpointAuthorization{}); err != nil {
Asim Shankarb547ea92015-02-17 18:49:45 -0800695 t.Errorf("Resolve should succeed when skipping server authorization. Got (%v, %v)", e, err)
Todd Wang8fa38762015-03-25 14:04:59 -0700696 } else if e, err := resolve("mt/server"); verror.ErrorID(err) != verror.ErrNotTrusted.ID {
Asim Shankarb547ea92015-02-17 18:49:45 -0800697 t.Errorf("Resolve should have failed with %q because an attacker has taken over the intermediate mounttable. Got (%+v, errorid=%q:%v)", verror.ErrNotTrusted.ID, e, verror.ErrorID(err), err)
698 }
699}
David Why Use Two When One Will Do Presotto18be8b02015-03-02 17:26:25 -0800700
701// TestDelete tests deleting some parts of the name space.
702func TestDelete(t *testing.T) {
703 _, c, cleanup := createContexts(t)
704 defer cleanup()
705 ns := v23.GetNamespace(c)
706
707 // Create a root mount table with mount tables mounted at mt1, mt1, ...
708 root, _, _, stopper := createNamespace(t, c)
709 defer stopper()
710 ns.SetRoots(root.name)
711
712 // We should be able to remove servers below the root.
713 if err := ns.Delete(c, "mt1", false); err != nil {
714 t.Errorf("Delete failed: %s", err)
715 }
716
717 // Create a server below one level down.
718 if err := ns.Mount(c, "mt2/b/c", "/madeup:1111/server", time.Minute); err != nil {
719 t.Errorf("Mount mt2/b/c failed: %s", err)
720 }
721
722 // We should not be able to delete mt2/b...
723 if err := ns.Delete(c, "mt2/b", false); err == nil {
724 t.Errorf("Delete mt2/b should have failed")
725 }
726
727 // ...unless we include its children.
728 if err := ns.Delete(c, "mt2/b", true); err != nil {
729 t.Errorf("Delete failed: %s", err)
730 }
731}
Robin Thellend89e95232015-03-24 13:48:48 -0700732
733type leafObject struct{}
734
735func (leafObject) Foo(rpc.ServerCall) error {
736 return nil
737}
738
739func TestLeaf(t *testing.T) {
740 _, ctx, cleanup := createContexts(t)
741 defer cleanup()
742 root := runMT(t, ctx, "")
743 ns := v23.GetNamespace(ctx)
744 ns.SetRoots(root.name)
745
746 server, err := v23.NewServer(ctx)
747 if err != nil {
748 boom(t, "v23.NewServer: %s", err)
749 }
750 ls := rpc.ListenSpec{Addrs: rpc.ListenAddrs{{"tcp", "127.0.0.1:0"}}}
751 if _, err := server.Listen(ls); err != nil {
752 boom(t, "Failed to Listen: %s", err)
753 }
754 if err := server.Serve("leaf", &leafObject{}, nil); err != nil {
755 boom(t, "server.Serve failed: %s", err)
756 }
757
758 mountEntry, err := ns.Resolve(ctx, "leaf")
759 if err != nil {
760 boom(t, "ns.Resolve failed: %v", err)
761 }
762 if expected := true; mountEntry.IsLeaf != expected {
763 boom(t, "unexpected mountEntry.IsLeaf value. Got %v, expected %v", mountEntry.IsLeaf, expected)
764 }
Robin Thellend00643742015-04-01 10:36:50 -0700765
766 c, err := ns.Glob(ctx, "leaf")
767 if err != nil {
768 boom(t, "ns.Glob failed: %v", err)
769 }
770 count := 0
771 for result := range c {
James Ringc971dca2015-04-09 09:51:20 -0700772 if me, ok := result.(*naming.GlobReplyEntry); ok {
Robin Thellend00643742015-04-01 10:36:50 -0700773 count++
James Ringc971dca2015-04-09 09:51:20 -0700774 if expected := true; me.Value.IsLeaf != expected {
775 boom(t, "unexpected me.IsLeaf value. Got %v, expected %v", me.Value.IsLeaf, expected)
Robin Thellend00643742015-04-01 10:36:50 -0700776 }
777 }
778 }
779 if count == 0 {
780 boom(t, "Glob did not return any results. Expected 1")
781 }
Robin Thellend89e95232015-03-24 13:48:48 -0700782}