blob: f57b00c8459a2ef80122d93a2389204b1391e3fd [file] [log] [blame]
Cosmos Nicolaou4e029972014-06-13 14:53:08 -07001package namespace_test
Jiri Simsa5293dcb2014-05-10 09:56:38 -07002
3import (
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -07004 "runtime"
Jiri Simsa5293dcb2014-05-10 09:56:38 -07005 "runtime/debug"
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -07006 "sync"
Jiri Simsa5293dcb2014-05-10 09:56:38 -07007 "testing"
8 "time"
9
Jiri Simsa764efb72014-12-25 20:57:03 -080010 "v.io/core/veyron2"
11 "v.io/core/veyron2/context"
12 "v.io/core/veyron2/ipc"
13 "v.io/core/veyron2/naming"
14 "v.io/core/veyron2/options"
Jiri Simsa764efb72014-12-25 20:57:03 -080015 "v.io/core/veyron2/security"
16 verror "v.io/core/veyron2/verror2"
17 "v.io/core/veyron2/vlog"
Cosmos Nicolaouf889c732014-10-16 20:46:54 -070018
Jiri Simsa764efb72014-12-25 20:57:03 -080019 "v.io/core/veyron/lib/testutil"
Ankurfb9255a2015-01-21 15:29:38 -080020 tsecurity "v.io/core/veyron/lib/testutil/security"
Suharsh Sivakumard68949c2015-01-26 10:32:23 -080021 _ "v.io/core/veyron/profiles"
Jiri Simsa764efb72014-12-25 20:57:03 -080022 "v.io/core/veyron/runtimes/google/naming/namespace"
23 vsecurity "v.io/core/veyron/security"
24 service "v.io/core/veyron/services/mounttable/lib"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070025)
26
Shyam Jayaramandbae76b2014-11-17 12:51:29 -080027func init() {
28 testutil.Init()
Shyam Jayaramandbae76b2014-11-17 12:51:29 -080029}
Asim Shankarc920db32014-10-16 19:18:21 -070030
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080031func createContexts(t *testing.T) (sc, c *context.T, cleanup func()) {
Ankurf416ac52015-01-29 13:58:24 -080032 ctx, shutdown := testutil.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080033 var err error
34 if sc, err = veyron2.SetPrincipal(ctx, tsecurity.NewPrincipal("test-blessing")); err != nil {
Ankurfb9255a2015-01-21 15:29:38 -080035 t.Fatal(err)
36 }
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080037 if c, err = veyron2.SetPrincipal(ctx, tsecurity.NewPrincipal("test-blessing")); err != nil {
Ankurfb9255a2015-01-21 15:29:38 -080038 t.Fatal(err)
39 }
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -080040 return sc, c, shutdown
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -080041}
42
Jiri Simsa5293dcb2014-05-10 09:56:38 -070043func boom(t *testing.T, f string, v ...interface{}) {
44 t.Logf(f, v...)
45 t.Fatal(string(debug.Stack()))
46}
47
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -070048// N squared but who cares, this is a little test.
49// Ignores dups.
50func contains(container, contained []string) bool {
51L:
52 for _, d := range contained {
53 for _, r := range container {
54 if r == d {
55 continue L
56 }
57 }
58 return false
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -070059 }
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -070060 return true
61}
62
63func compare(t *testing.T, caller, name string, got, want []string) {
64 // Compare ignoring dups.
65 if !contains(got, want) || !contains(want, got) {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -070066 boom(t, "%s: %q: got %v, want %v", caller, name, got, want)
67 }
68}
69
Matt Rosencrantz6edab562015-01-12 11:07:55 -080070func doGlob(t *testing.T, ctx *context.T, ns naming.Namespace, pattern string, limit int) []string {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070071 var replies []string
Matt Rosencrantz6edab562015-01-12 11:07:55 -080072 rc, err := ns.Glob(ctx, pattern)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070073 if err != nil {
74 boom(t, "Glob(%s): %s", pattern, err)
75 }
76 for s := range rc {
77 replies = append(replies, s.Name)
David Why Use Two When One Will Do Presotto2db32f92014-07-18 12:20:35 -070078 if limit > 0 && len(replies) > limit {
79 boom(t, "Glob returns too many results, perhaps not limiting recursion")
80 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -070081 }
82 return replies
83}
84
Robin Thellend434e39f2014-08-27 14:46:27 -070085type testServer struct {
86 suffix string
87}
Jiri Simsa5293dcb2014-05-10 09:56:38 -070088
Todd Wang1fe7cdd2014-11-12 12:51:49 -080089func (testServer) KnockKnock(ctx ipc.ServerContext) string {
Jiri Simsa5293dcb2014-05-10 09:56:38 -070090 return "Who's there?"
91}
92
Robin Thellend39ac3232014-12-02 09:50:41 -080093// testServer has the following namespace:
Robin Thellend434e39f2014-08-27 14:46:27 -070094// "" -> {level1} -> {level2}
Robin Thellend39ac3232014-12-02 09:50:41 -080095func (t *testServer) GlobChildren__(ipc.ServerContext) (<-chan string, error) {
96 ch := make(chan string, 1)
97 switch t.suffix {
98 case "":
99 ch <- "level1"
100 case "level1":
101 ch <- "level2"
102 default:
103 return nil, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700104 }
Robin Thellend39ac3232014-12-02 09:50:41 -0800105 close(ch)
106 return ch, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700107}
108
Asim Shankar98d65962014-10-24 16:55:30 -0700109type allowEveryoneAuthorizer struct{}
110
111func (allowEveryoneAuthorizer) Authorize(security.Context) error { return nil }
112
Robin Thellend434e39f2014-08-27 14:46:27 -0700113type dispatcher struct{}
114
Robin Thellenda02fe8f2014-11-19 09:58:29 -0800115func (d *dispatcher) Lookup(suffix string) (interface{}, security.Authorizer, error) {
Cosmos Nicolaou710daa22014-11-11 19:39:18 -0800116 return &testServer{suffix}, allowEveryoneAuthorizer{}, nil
Robin Thellend434e39f2014-08-27 14:46:27 -0700117}
118
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800119func knockKnock(t *testing.T, ctx *context.T, name string) {
120 client := veyron2.GetClient(ctx)
Matt Rosencrantzf5afcaf2014-06-02 11:31:22 -0700121 call, err := client.StartCall(ctx, name, "KnockKnock", nil)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700122 if err != nil {
123 boom(t, "StartCall failed: %s", err)
124 }
125 var result string
126 if err := call.Finish(&result); err != nil {
127 boom(t, "Finish returned an error: %s", err)
128 }
129 if result != "Who's there?" {
130 boom(t, "Wrong result: %v", result)
131 }
132}
133
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800134func doResolveTest(t *testing.T, fname string, f func(*context.T, string, ...naming.ResolveOpt) (*naming.MountEntry, error), ctx *context.T, name string, want []string, opts ...naming.ResolveOpt) {
135 me, err := f(ctx, name, opts...)
Ryan Brownbc2c87c2014-11-17 18:55:25 +0000136 if err != nil {
Ryan Brownfc290142014-11-24 15:47:24 -0800137 boom(t, "Failed to %s %s: %s", fname, name, err)
Ryan Brownbc2c87c2014-11-17 18:55:25 +0000138 }
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800139 compare(t, fname, name, me.Names(), want)
Ryan Brownfc290142014-11-24 15:47:24 -0800140}
141
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800142func testResolveToMountTable(t *testing.T, ctx *context.T, ns naming.Namespace, name string, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800143 doResolveTest(t, "ResolveToMountTable", ns.ResolveToMountTable, ctx, name, want)
Ryan Brownfc290142014-11-24 15:47:24 -0800144}
145
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800146func testResolveToMountTableWithPattern(t *testing.T, ctx *context.T, ns naming.Namespace, name string, pattern naming.ResolveOpt, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800147 doResolveTest(t, "ResolveToMountTable", ns.ResolveToMountTable, ctx, name, want, pattern)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700148}
149
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800150func testResolve(t *testing.T, ctx *context.T, ns naming.Namespace, name string, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800151 doResolveTest(t, "Resolve", ns.Resolve, ctx, name, want)
Ryan Brownfc290142014-11-24 15:47:24 -0800152}
153
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800154func testResolveWithPattern(t *testing.T, ctx *context.T, ns naming.Namespace, name string, pattern naming.ResolveOpt, want ...string) {
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800155 doResolveTest(t, "Resolve", ns.Resolve, ctx, name, want, pattern)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700156}
157
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700158type serverEntry struct {
159 mountPoint string
160 server ipc.Server
161 endpoint naming.Endpoint
162 name string
163}
164
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800165func runServer(t *testing.T, ctx *context.T, disp ipc.Dispatcher, mountPoint string) *serverEntry {
166 return run(t, ctx, disp, mountPoint, false)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700167}
168
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800169func runMT(t *testing.T, ctx *context.T, mountPoint string) *serverEntry {
Bogdan Capritae96cd042015-02-03 17:32:57 -0800170 mtd, err := service.NewMountTableDispatcher("")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700171 if err != nil {
Bogdan Capritae96cd042015-02-03 17:32:57 -0800172 boom(t, "NewMountTableDispatcher returned error: %v", err)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700173 }
Bogdan Capritae96cd042015-02-03 17:32:57 -0800174 return run(t, ctx, mtd, mountPoint, true)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700175}
176
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800177func run(t *testing.T, ctx *context.T, disp ipc.Dispatcher, mountPoint string, mt bool) *serverEntry {
178 s, err := veyron2.NewServer(ctx, options.ServesMountTable(mt))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700179 if err != nil {
180 boom(t, "r.NewServer: %s", err)
181 }
182 // Add a mount table server.
183 // Start serving on a loopback address.
Suharsh Sivakumard68949c2015-01-26 10:32:23 -0800184 eps, err := s.Listen(veyron2.GetListenSpec(ctx))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700185 if err != nil {
186 boom(t, "Failed to Listen: %s", err)
187 }
Cosmos Nicolaou92dba582014-11-05 17:24:10 -0800188 if err := s.ServeDispatcher(mountPoint, disp); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700189 boom(t, "Failed to serve mount table at %s: %s", mountPoint, err)
190 }
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800191 return &serverEntry{mountPoint: mountPoint, server: s, endpoint: eps[0], name: eps[0].Name()}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700192}
193
194const (
195 mt1MP = "mt1"
196 mt2MP = "mt2"
197 mt3MP = "mt3"
198 mt4MP = "mt4"
199 mt5MP = "mt5"
200 j1MP = "joke1"
201 j2MP = "joke2"
202 j3MP = "joke3"
203
204 ttl = 100 * time.Second
205)
206
207// runMountTables creates a root mountable with some mount tables mounted
208// in it: mt{1,2,3,4,5}
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800209func runMountTables(t *testing.T, ctx *context.T) (*serverEntry, map[string]*serverEntry) {
210 root := runMT(t, ctx, "")
211 veyron2.GetNamespace(ctx).SetRoots(root.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700212 t.Logf("mountTable %q -> %s", root.mountPoint, root.endpoint)
213
214 mps := make(map[string]*serverEntry)
215 for _, mp := range []string{mt1MP, mt2MP, mt3MP, mt4MP, mt5MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800216 m := runMT(t, ctx, mp)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700217 t.Logf("mountTable %q -> %s", mp, m.endpoint)
218 mps[mp] = m
219 }
220 return root, mps
221}
222
Robin Thellend434e39f2014-08-27 14:46:27 -0700223// createNamespace creates a hierarchy of mounttables and servers
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700224// as follows:
225// /mt1, /mt2, /mt3, /mt4, /mt5, /joke1, /joke2, /joke3.
226// That is, mt1 is a mount table mounted in the root mount table,
227// joke1 is a server mounted in the root mount table.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800228func createNamespace(t *testing.T, ctx *context.T) (*serverEntry, map[string]*serverEntry, map[string]*serverEntry, func()) {
229 root, mts := runMountTables(t, ctx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700230 jokes := make(map[string]*serverEntry)
231 // Let's run some non-mount table services.
232 for _, j := range []string{j1MP, j2MP, j3MP} {
Robin Thellend434e39f2014-08-27 14:46:27 -0700233 disp := &dispatcher{}
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800234 jokes[j] = runServer(t, ctx, disp, j)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700235 }
236 return root, mts, jokes, func() {
237 for _, s := range jokes {
238 s.server.Stop()
239 }
240 for _, s := range mts {
241 s.server.Stop()
242 }
243 root.server.Stop()
244 }
245}
246
247// runNestedMountTables creates some nested mount tables in the hierarchy
248// created by createNamespace as follows:
249// /mt4/foo, /mt4/foo/bar and /mt4/baz where foo, bar and baz are mount tables.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800250func runNestedMountTables(t *testing.T, ctx *context.T, mts map[string]*serverEntry) {
251 ns := veyron2.GetNamespace(ctx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700252 // Set up some nested mounts and verify resolution.
253 for _, m := range []string{"mt4/foo", "mt4/foo/bar"} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800254 mts[m] = runMT(t, ctx, m)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700255 }
256
257 // Use a global name for a mount, rather than a relative one.
258 // We directly mount baz into the mt4/foo mount table.
259 globalMP := naming.JoinAddressName(mts["mt4/foo"].name, "baz")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800260 mts["baz"] = runMT(t, ctx, "baz")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700261 if err := ns.Mount(ctx, globalMP, mts["baz"].name, ttl); err != nil {
262 boom(t, "Failed to Mount %s: %s", globalMP, err)
263 }
264}
265
266// TestNamespaceCommon tests common use of the Namespace library
267// against a root mount table and some mount tables mounted on it.
268func TestNamespaceCommon(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800269 _, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800270 defer cleanup()
271
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800272 root, mts, jokes, stopper := createNamespace(t, c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700273 defer stopper()
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800274 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700275
276 // All of the initial mounts are served by the root mounttable
277 // and hence ResolveToMountTable should return the root mountable
278 // as the address portion of the terminal name for those mounttables.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800279 testResolveToMountTable(t, c, ns, "", root.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700280 for _, m := range []string{mt2MP, mt3MP, mt5MP} {
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800281 rootMT := naming.Join(root.name, m)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700282 // All of these mount tables are hosted by the root mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800283 testResolveToMountTable(t, c, ns, m, rootMT)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700284
285 // The server registered for each mount point is a mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800286 testResolve(t, c, ns, m, mts[m].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700287
288 // ResolveToMountTable will walk through to the sub MountTables
289 mtbar := naming.Join(m, "bar")
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800290 subMT := naming.Join(mts[m].name, "bar")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800291 testResolveToMountTable(t, c, ns, mtbar, subMT)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700292 }
293
294 for _, j := range []string{j1MP, j2MP, j3MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800295 testResolve(t, c, ns, j, jokes[j].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700296 }
297}
298
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800299// TestNamespaceDetails tests more detailed use of the Namespace library.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700300func TestNamespaceDetails(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800301 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800302 defer cleanup()
303
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800304 root, mts, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700305 defer stopper()
306
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800307 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700308 ns.SetRoots(root.name)
309
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800310 // /mt2 is not an endpoint. Thus, the example below will fail.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700311 mt3Server := mts[mt3MP].name
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800312 mt2a := "/mt2/a"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800313 if err := ns.Mount(c, mt2a, mt3Server, ttl); verror.Is(err, naming.ErrNoSuchName.ID) {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700314 boom(t, "Successfully mounted %s - expected an err %v, not %v", mt2a, naming.ErrNoSuchName, err)
315 }
316
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800317 // Mount using the relative name.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700318 // This means walk through mt2 if it already exists and mount within
319 // the lower level mount table, if the name doesn't exist we'll create
320 // a new name for it.
321 mt2a = "mt2/a"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800322 if err := ns.Mount(c, mt2a, mt3Server, ttl); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700323 boom(t, "Failed to Mount %s: %s", mt2a, err)
324 }
325
David Why Use Two When One Will Do Presotto8b4dbbf2014-11-06 10:50:14 -0800326 mt2mt := naming.Join(mts[mt2MP].name, "a")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700327 // The mt2/a is served by the mt2 mount table
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800328 testResolveToMountTable(t, c, ns, mt2a, mt2mt)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700329 // The server for mt2a is mt3server from the second mount above.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800330 testResolve(t, c, ns, mt2a, mt3Server)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700331
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800332 // Add two more mounts. The // should be stripped off of the
333 // second.
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700334 for _, mp := range []struct{ name, server string }{
335 {"mt2", mts[mt4MP].name},
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800336 {"mt2//", mts[mt5MP].name},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700337 } {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800338 if err := ns.Mount(c, mp.name, mp.server, ttl, naming.ServesMountTableOpt(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700339 boom(t, "Failed to Mount %s: %s", mp.name, err)
340 }
341 }
342
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800343 names := []string{naming.JoinAddressName(mts[mt4MP].name, "a"),
344 naming.JoinAddressName(mts[mt5MP].name, "a")}
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800345 names = append(names, naming.JoinAddressName(mts[mt2MP].name, "a"))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700346 // We now have 3 mount tables prepared to serve mt2/a
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800347 testResolveToMountTable(t, c, ns, "mt2/a", names...)
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800348 names = []string{mts[mt4MP].name, mts[mt5MP].name}
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800349 names = append(names, mts[mt2MP].name)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800350 testResolve(t, c, ns, "mt2", names...)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700351}
352
353// TestNestedMounts tests some more deeply nested mounts
354func TestNestedMounts(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800355 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800356 defer cleanup()
357
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800358 root, mts, _, stopper := createNamespace(t, sc)
359 runNestedMountTables(t, sc, mts)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700360 defer stopper()
361
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800362 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700363 ns.SetRoots(root.name)
364
365 // Set up some nested mounts and verify resolution.
366 for _, m := range []string{"mt4/foo", "mt4/foo/bar"} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800367 testResolve(t, c, ns, m, mts[m].name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700368 }
369
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800370 testResolveToMountTable(t, c, ns, "mt4/foo",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800371 naming.JoinAddressName(mts[mt4MP].name, "foo"))
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800372 testResolveToMountTable(t, c, ns, "mt4/foo/bar",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800373 naming.JoinAddressName(mts["mt4/foo"].name, "bar"))
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800374 testResolveToMountTable(t, c, ns, "mt4/foo/baz",
Cosmos Nicolaouae8dd212014-12-13 23:43:08 -0800375 naming.JoinAddressName(mts["mt4/foo"].name, "baz"))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700376}
377
378// TestServers tests invoking RPCs on simple servers
379func TestServers(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800380 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800381 defer cleanup()
382
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800383 root, mts, jokes, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700384 defer stopper()
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800385 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700386 ns.SetRoots(root.name)
387
Robin Thellend434e39f2014-08-27 14:46:27 -0700388 // Let's run some non-mount table services
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700389 for _, j := range []string{j1MP, j2MP, j3MP} {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800390 testResolve(t, c, ns, j, jokes[j].name)
391 knockKnock(t, c, j)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700392 globalName := naming.JoinAddressName(mts["mt4"].name, j)
Robin Thellend434e39f2014-08-27 14:46:27 -0700393 disp := &dispatcher{}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700394 gj := "g_" + j
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800395 jokes[gj] = runServer(t, c, disp, globalName)
396 testResolve(t, c, ns, "mt4/"+j, jokes[gj].name)
397 knockKnock(t, c, "mt4/"+j)
398 testResolveToMountTable(t, c, ns, "mt4/"+j, globalName)
399 testResolveToMountTable(t, c, ns, "mt4/"+j+"/garbage", globalName+"/garbage")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700400 }
401}
402
403// TestGlob tests some glob patterns.
404func TestGlob(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800405 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800406 defer cleanup()
407
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800408 root, mts, _, stopper := createNamespace(t, sc)
409 runNestedMountTables(t, sc, mts)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700410 defer stopper()
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800411 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700412 ns.SetRoots(root.name)
413
414 tln := []string{"baz", "mt1", "mt2", "mt3", "mt4", "mt5", "joke1", "joke2", "joke3"}
415 barbaz := []string{"mt4/foo/bar", "mt4/foo/baz"}
Robin Thellend434e39f2014-08-27 14:46:27 -0700416 level12 := []string{"joke1/level1", "joke1/level1/level2", "joke2/level1", "joke2/level1/level2", "joke3/level1", "joke3/level1/level2"}
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700417 foo := append([]string{"mt4/foo"}, barbaz...)
Robin Thellend434e39f2014-08-27 14:46:27 -0700418 foo = append(foo, level12...)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700419 // Try various globs.
420 globTests := []struct {
421 pattern string
422 expected []string
423 }{
424 {"*", tln},
David Why Use Two When One Will Do Presottocce8f4f2014-09-30 14:50:44 -0700425 {"x", []string{}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700426 {"m*", []string{"mt1", "mt2", "mt3", "mt4", "mt5"}},
427 {"mt[2,3]", []string{"mt2", "mt3"}},
428 {"*z", []string{"baz"}},
Robin Thellend434e39f2014-08-27 14:46:27 -0700429 {"joke1/*", []string{"joke1/level1"}},
430 {"j?ke1/level1/*", []string{"joke1/level1/level2"}},
431 {"joke1/level1/*", []string{"joke1/level1/level2"}},
432 {"joke1/level1/level2/...", []string{"joke1/level1/level2"}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700433 {"...", append(append(tln, foo...), "")},
434 {"*/...", append(tln, foo...)},
435 {"*/foo/*", barbaz},
436 {"*/*/*z", []string{"mt4/foo/baz"}},
437 {"*/f??/*z", []string{"mt4/foo/baz"}},
Robin Thellend434e39f2014-08-27 14:46:27 -0700438 {"mt4/foo/baz", []string{"mt4/foo/baz"}},
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700439 }
440 for _, test := range globTests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800441 out := doGlob(t, c, ns, test.pattern, 0)
Robin Thellend434e39f2014-08-27 14:46:27 -0700442 compare(t, "Glob", test.pattern, out, test.expected)
Robin Thellend5f838572014-07-10 13:28:56 -0700443 // Do the same with a full rooted name.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800444 out = doGlob(t, c, ns, naming.JoinAddressName(root.name, test.pattern), 0)
Robin Thellend5f838572014-07-10 13:28:56 -0700445 var expectedWithRoot []string
446 for _, s := range test.expected {
447 expectedWithRoot = append(expectedWithRoot, naming.JoinAddressName(root.name, s))
448 }
Robin Thellend434e39f2014-08-27 14:46:27 -0700449 compare(t, "Glob", test.pattern, out, expectedWithRoot)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700450 }
451}
452
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700453type GlobbableServer struct {
454 callCount int
455 mu sync.Mutex
456}
457
Robin Thellend39ac3232014-12-02 09:50:41 -0800458func (g *GlobbableServer) Glob__(ipc.ServerContext, string) (<-chan naming.VDLMountEntry, error) {
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700459 g.mu.Lock()
460 defer g.mu.Unlock()
461 g.callCount++
Robin Thellend39ac3232014-12-02 09:50:41 -0800462 return nil, nil
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700463}
464
465func (g *GlobbableServer) GetAndResetCount() int {
466 g.mu.Lock()
467 defer g.mu.Unlock()
468 cnt := g.callCount
469 g.callCount = 0
470
471 return cnt
472}
473
474// TestGlobEarlyStop tests that Glob doesn't query terminal servers with finished patterns.
475func TestGlobEarlyStop(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800476 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800477 defer cleanup()
478
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800479 root, mts, _, stopper := createNamespace(t, sc)
480 runNestedMountTables(t, sc, mts)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700481 defer stopper()
482
483 globServer := &GlobbableServer{}
484 name := naming.JoinAddressName(mts["mt4/foo/bar"].name, "glob")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800485 runningGlobServer := runServer(t, c, testutil.LeafDispatcher(globServer, nil), name)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700486 defer runningGlobServer.server.Stop()
487
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800488 ns := veyron2.GetNamespace(c)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700489 ns.SetRoots(root.name)
490
491 tests := []struct {
492 pattern string
493 expectedCalls int
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800494 expected []string
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700495 }{
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800496 {"mt4/foo/bar/glob", 0, []string{"mt4/foo/bar/glob"}},
497 {"mt4/foo/bar/glob/...", 1, []string{"mt4/foo/bar/glob"}},
498 {"mt4/foo/bar/glob/*", 1, nil},
499 {"mt4/foo/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
500 {"mt4/foo/bar/...", 1, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
501 {"mt4/foo/bar/*", 0, []string{"mt4/foo/bar/glob"}},
502 {"mt4/***/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
503 {"mt4/*/bar/***", 0, []string{"mt4/foo/bar", "mt4/foo/bar/glob"}},
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700504 }
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800505 // Test allowing the tests to descend into leaves.
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700506 for _, test := range tests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800507 out := doGlob(t, c, ns, test.pattern, 0)
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800508 compare(t, "Glob", test.pattern, out, test.expected)
Matt Rosencrantz50c2bb82014-07-24 09:10:33 -0700509 if calls := globServer.GetAndResetCount(); calls != test.expectedCalls {
510 boom(t, "Wrong number of Glob calls to terminal server got: %d want: %d.", calls, test.expectedCalls)
511 }
512 }
513}
514
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700515func TestCycles(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800516 sc, c, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800517 defer cleanup()
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700518
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800519 root, _, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700520 defer stopper()
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800521 ns := veyron2.GetNamespace(c)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700522 ns.SetRoots(root.name)
523
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800524 c1 := runMT(t, c, "c1")
525 c2 := runMT(t, c, "c2")
526 c3 := runMT(t, c, "c3")
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700527 defer c1.server.Stop()
528 defer c2.server.Stop()
529 defer c3.server.Stop()
530
531 m := "c1/c2"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800532 if err := ns.Mount(c, m, c1.name, ttl, naming.ServesMountTableOpt(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700533 boom(t, "Failed to Mount %s: %s", "c1/c2", err)
534 }
535
536 m = "c1/c2/c3"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800537 if err := ns.Mount(c, m, c3.name, ttl, naming.ServesMountTableOpt(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700538 boom(t, "Failed to Mount %s: %s", m, err)
539 }
540
541 m = "c1/c3/c4"
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800542 if err := ns.Mount(c, m, c1.name, ttl, naming.ServesMountTableOpt(true)); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700543 boom(t, "Failed to Mount %s: %s", m, err)
544 }
545
Shyam Jayaramandbae76b2014-11-17 12:51:29 -0800546 // Since c1 was mounted with the Serve call, it will have both the tcp and ws endpoints.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800547 testResolve(t, c, ns, "c1", c1.name)
548 testResolve(t, c, ns, "c1/c2", c1.name)
549 testResolve(t, c, ns, "c1/c3", c3.name)
550 testResolve(t, c, ns, "c1/c3/c4", c1.name)
551 testResolve(t, c, ns, "c1/c3/c4/c3/c4", c1.name)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700552 cycle := "c3/c4"
553 for i := 0; i < 40; i++ {
554 cycle += "/c3/c4"
555 }
David Why Use Two When One Will Do Presotto8de85852015-01-21 11:05:09 -0800556 if _, err := ns.Resolve(c, "c1/"+cycle); !verror.Is(err, naming.ErrResolutionDepthExceeded.ID) {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700557 boom(t, "Failed to detect cycle")
558 }
559
David Why Use Two When One Will Do Presotto2db32f92014-07-18 12:20:35 -0700560 // Perform the glob with a response length limit.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800561 doGlob(t, c, ns, "c1/...", 1000)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700562}
563
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700564// TestGoroutineLeaks tests for leaking goroutines - we have many:-(
565func TestGoroutineLeaks(t *testing.T) {
566 t.Skip()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800567 sc, _, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800568 defer cleanup()
569
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800570 _, _, _, stopper := createNamespace(t, sc)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700571 defer func() {
572 vlog.Infof("%d goroutines:", runtime.NumGoroutine())
573 }()
574 defer stopper()
575 defer func() {
576 vlog.Infof("%d goroutines:", runtime.NumGoroutine())
577 }()
578 //panic("this will show up lots of goroutine+channel leaks!!!!")
579}
580
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700581func TestBadRoots(t *testing.T) {
Matt Rosencrantzafe49642014-12-05 15:10:56 -0800582 if _, err := namespace.New(); err != nil {
Cosmos Nicolaou4e029972014-06-13 14:53:08 -0700583 t.Errorf("namespace.New should not have failed with no roots")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700584 }
Matt Rosencrantzafe49642014-12-05 15:10:56 -0800585 if _, err := namespace.New("not a rooted name"); err == nil {
Cosmos Nicolaou4e029972014-06-13 14:53:08 -0700586 t.Errorf("namespace.New should have failed with an unrooted name")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700587 }
588}
Ryan Brownfc290142014-11-24 15:47:24 -0800589
590func bless(blesser, delegate security.Principal, extension string) {
591 b, err := blesser.Bless(delegate.PublicKey(), blesser.BlessingStore().Default(), extension, security.UnconstrainedUse())
592 if err != nil {
593 panic(err)
594 }
595 delegate.BlessingStore().SetDefault(b)
596}
597
598func TestRootBlessing(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800599 c, cc, cleanup := createContexts(t)
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800600 defer cleanup()
Ryan Brownfc290142014-11-24 15:47:24 -0800601
602 proot, err := vsecurity.NewPrincipal()
603 if err != nil {
604 panic(err)
605 }
606 b, err := proot.BlessSelf("root")
607 if err != nil {
608 panic(err)
609 }
610 proot.BlessingStore().SetDefault(b)
611
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800612 sprincipal := veyron2.GetPrincipal(c)
613 cprincipal := veyron2.GetPrincipal(cc)
614 bless(proot, sprincipal, "server")
615 bless(proot, cprincipal, "client")
616 cprincipal.AddToRoots(proot.BlessingStore().Default())
617 sprincipal.AddToRoots(proot.BlessingStore().Default())
Ryan Brownfc290142014-11-24 15:47:24 -0800618
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800619 root, mts, _, stopper := createNamespace(t, c)
Ryan Brownfc290142014-11-24 15:47:24 -0800620 defer stopper()
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800621 ns := veyron2.GetNamespace(c)
Ryan Brownfc290142014-11-24 15:47:24 -0800622
623 name := naming.Join(root.name, mt2MP)
624 // First check with a non-matching blessing pattern.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800625 _, err = ns.Resolve(c, name, naming.RootBlessingPatternOpt("root/foobar"))
Ryan Brownfc290142014-11-24 15:47:24 -0800626 if !verror.Is(err, verror.NotTrusted.ID) {
627 t.Errorf("Resolve expected NotTrusted error, got %v", err)
628 }
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800629 _, err = ns.ResolveToMountTable(c, name, naming.RootBlessingPatternOpt("root/foobar"))
Ryan Brownfc290142014-11-24 15:47:24 -0800630 if !verror.Is(err, verror.NotTrusted.ID) {
631 t.Errorf("ResolveToMountTable expected NotTrusted error, got %v", err)
632 }
633
634 // Now check a matching pattern.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800635 testResolveWithPattern(t, c, ns, name, naming.RootBlessingPatternOpt("root/server"), mts[mt2MP].name)
636 testResolveToMountTableWithPattern(t, c, ns, name, naming.RootBlessingPatternOpt("root/server"), name)
Ryan Brownfc290142014-11-24 15:47:24 -0800637
638 // After successful lookup it should be cached, so the pattern doesn't matter.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800639 testResolveWithPattern(t, c, ns, name, naming.RootBlessingPatternOpt("root/foobar"), mts[mt2MP].name)
Ryan Brownfc290142014-11-24 15:47:24 -0800640}