blob: 6d27b94f895660df386d38582f2d005ac817c6e4 [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
Todd Wang5987a942015-04-06 11:06:17 -07005package mounttablelib
Jiri Simsa5293dcb2014-05-10 09:56:38 -07006
7import (
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -07008 "encoding/json"
Jiri Simsa5293dcb2014-05-10 09:56:38 -07009 "errors"
Robin Thellendfd82dcc2015-04-14 21:16:59 -070010 "fmt"
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070011 "io"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070012 "reflect"
13 "runtime/debug"
14 "sort"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070015 "testing"
16 "time"
17
Jiri Simsa6ac95222015-02-23 16:11:49 -080018 "v.io/v23"
19 "v.io/v23/context"
Jiri Simsa6ac95222015-02-23 16:11:49 -080020 "v.io/v23/naming"
21 "v.io/v23/options"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070022 "v.io/v23/rpc"
Jiri Simsa6ac95222015-02-23 16:11:49 -080023 "v.io/v23/security"
Todd Wang387d8a42015-03-30 17:09:05 -070024 "v.io/v23/security/access"
Robin Thellendfd82dcc2015-04-14 21:16:59 -070025 "v.io/v23/services/stats"
26 "v.io/v23/vdl"
Jiri Simsa337af232015-02-27 14:36:46 -080027 "v.io/x/lib/vlog"
Cosmos Nicolaoud6c3c9c2014-09-30 15:42:53 -070028
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -070029 libstats "v.io/x/ref/lib/stats"
Robin Thellendfd82dcc2015-04-14 21:16:59 -070030 "v.io/x/ref/services/debug/debuglib"
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -070031 "v.io/x/ref/test"
Asim Shankar4a698282015-03-21 21:59:18 -070032 "v.io/x/ref/test/testutil"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070033)
34
Asim Shankar88292912014-10-09 19:41:07 -070035// Simulate different processes with different runtimes.
Matt Rosencrantz6edab562015-01-12 11:07:55 -080036// rootCtx is the one running the mounttable service.
Jiri Simsa5293dcb2014-05-10 09:56:38 -070037const ttlSecs = 60 * 60
38
39func boom(t *testing.T, f string, v ...interface{}) {
40 t.Logf(f, v...)
41 t.Fatal(string(debug.Stack()))
42}
43
Asim Shankar43d1f932015-03-24 20:57:56 -070044func doMount(t *testing.T, ctx *context.T, ep, suffix, service string, shouldSucceed bool) {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070045 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -080046 client := v23.GetClient(ctx)
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -070047 if err := client.Call(ctx, name, "Mount", []interface{}{service, uint32(ttlSecs), 0}, nil, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070048 if !shouldSucceed {
49 return
50 }
51 boom(t, "Failed to Mount %s onto %s: %s", service, name, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070052 }
53}
54
Matt Rosencrantz6edab562015-01-12 11:07:55 -080055func doUnmount(t *testing.T, ctx *context.T, ep, suffix, service string, shouldSucceed bool) {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070056 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -080057 client := v23.GetClient(ctx)
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -070058 if err := client.Call(ctx, name, "Unmount", []interface{}{service}, nil, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070059 if !shouldSucceed {
60 return
61 }
Alex Fandriantoa2e7c372015-04-20 10:46:10 -070062 boom(t, "Failed to Unmount %s off of %s: %s", service, name, err)
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -070063 }
64}
65
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -070066func doGetPermissions(t *testing.T, ctx *context.T, ep, suffix string, shouldSucceed bool) (perms access.Permissions, version string) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080067 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -080068 client := v23.GetClient(ctx)
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -070069 if err := client.Call(ctx, name, "GetPermissions", nil, []interface{}{&perms, &version}, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080070 if !shouldSucceed {
71 return
72 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070073 boom(t, "Failed to GetPermissions %s: %s", name, err)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080074 }
75 return
76}
77
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -070078func doSetPermissions(t *testing.T, ctx *context.T, ep, suffix string, perms access.Permissions, version string, shouldSucceed bool) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080079 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -080080 client := v23.GetClient(ctx)
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -070081 if err := client.Call(ctx, name, "SetPermissions", []interface{}{perms, version}, nil, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080082 if !shouldSucceed {
83 return
84 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070085 boom(t, "Failed to SetPermissions %s: %s", name, err)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080086 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080087}
88
Matt Rosencrantz6edab562015-01-12 11:07:55 -080089func doDeleteNode(t *testing.T, ctx *context.T, ep, suffix string, shouldSucceed bool) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080090 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -080091 client := v23.GetClient(ctx)
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -070092 if err := client.Call(ctx, name, "Delete", []interface{}{false}, nil, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080093 if !shouldSucceed {
94 return
95 }
96 boom(t, "Failed to Delete node %s: %s", name, err)
97 }
98}
99
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800100func doDeleteSubtree(t *testing.T, ctx *context.T, ep, suffix string, shouldSucceed bool) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800101 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -0800102 client := v23.GetClient(ctx)
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -0700103 if err := client.Call(ctx, name, "Delete", []interface{}{true}, nil, options.NoResolve{}); err != nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800104 if !shouldSucceed {
105 return
106 }
107 boom(t, "Failed to Delete subtree %s: %s", name, err)
108 }
109}
110
Todd Wang2331dd02015-03-17 15:38:39 -0700111func mountentry2names(e *naming.MountEntry) []string {
Asim Shankar8bc78d62015-02-03 16:53:27 -0800112 names := make([]string, len(e.Servers))
113 for idx, s := range e.Servers {
114 names[idx] = naming.JoinAddressName(s.Server, e.Name)
115 }
116 return names
117}
118
119func strslice(strs ...string) []string {
120 return strs
121}
122
Todd Wang2331dd02015-03-17 15:38:39 -0700123func resolve(ctx *context.T, name string) (*naming.MountEntry, error) {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700124 // Resolve the name one level.
Todd Wang2331dd02015-03-17 15:38:39 -0700125 var entry naming.MountEntry
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -0700126 client := v23.GetClient(ctx)
127 if err := client.Call(ctx, name, "ResolveStep", nil, []interface{}{&entry}, options.NoResolve{}); err != nil {
Todd Wange77f9952015-02-18 13:20:50 -0800128 return nil, err
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700129 }
130 if len(entry.Servers) < 1 {
Asim Shankar8bc78d62015-02-03 16:53:27 -0800131 return nil, errors.New("resolve returned no servers")
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700132 }
Asim Shankar8bc78d62015-02-03 16:53:27 -0800133 return &entry, nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700134}
135
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800136func export(t *testing.T, ctx *context.T, name, contents string) {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700137 // Resolve the name.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800138 resolved, err := resolve(ctx, name)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700139 if err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700140 boom(t, "Failed to Export.Resolve %s: %s", name, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700141 }
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700142 // Export the value.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800143 client := v23.GetClient(ctx)
Suharsh Sivakumar1abd5a82015-04-10 00:24:14 -0700144 if err := client.Call(ctx, mountentry2names(resolved)[0], "Export", []interface{}{contents, true}, nil, options.NoResolve{}); err != nil {
145 boom(t, "Failed to Export.Call %s to %s: %s", name, contents, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700146 }
147}
148
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800149func checkContents(t *testing.T, ctx *context.T, name, expected string, shouldSucceed bool) {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700150 // Resolve the name.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800151 resolved, err := resolve(ctx, name)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700152 if err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700153 if !shouldSucceed {
154 return
155 }
156 boom(t, "Failed to Resolve %s: %s", name, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700157 }
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700158 // Look up the value.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800159 client := v23.GetClient(ctx)
Asim Shankar8bc78d62015-02-03 16:53:27 -0800160 call, err := client.StartCall(ctx, mountentry2names(resolved)[0], "Lookup", nil, options.NoResolve{})
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700161 if err != nil {
162 if shouldSucceed {
163 boom(t, "Failed Lookup.StartCall %s: %s", name, err)
164 }
165 return
166 }
167 var contents []byte
Todd Wange77f9952015-02-18 13:20:50 -0800168 if err := call.Finish(&contents); err != nil {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700169 if shouldSucceed {
170 boom(t, "Failed to Lookup %s: %s", name, err)
171 }
172 return
173 }
174 if string(contents) != expected {
175 boom(t, "Lookup %s, expected %q, got %q", name, expected, contents)
176 }
177 if !shouldSucceed {
178 boom(t, "Lookup %s, expected failure, got %q", name, contents)
179 }
180}
181
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700182func newMT(t *testing.T, permsFile, persistDir, statsDir string, rootCtx *context.T) (rpc.Server, string) {
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700183 reservedDisp := debuglib.NewDispatcher(vlog.Log.LogDir, nil)
Todd Wangad492042015-04-17 15:58:40 -0700184 ctx := v23.WithReservedNameDispatcher(rootCtx, reservedDisp)
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700185 server, err := v23.NewServer(ctx, options.ServesMountTable(true))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700186 if err != nil {
187 boom(t, "r.NewServer: %s", err)
188 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700189 // Add mount table service.
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700190 mt, err := NewMountTableDispatcher(permsFile, persistDir, statsDir)
Ryan Brownac972652014-05-19 10:57:32 -0700191 if err != nil {
Bogdan Capritae96cd042015-02-03 17:32:57 -0800192 boom(t, "NewMountTableDispatcher: %v", err)
Ryan Brownac972652014-05-19 10:57:32 -0700193 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700194 // Start serving on a loopback address.
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700195 eps, err := server.Listen(v23.GetListenSpec(ctx))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700196 if err != nil {
197 boom(t, "Failed to Listen mount table: %s", err)
198 }
Cosmos Nicolaou92dba582014-11-05 17:24:10 -0800199 if err := server.ServeDispatcher("", mt); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700200 boom(t, "Failed to register mock collection: %s", err)
201 }
Cosmos Nicolaou28dabfc2014-12-15 22:51:07 -0800202 estr := eps[0].String()
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700203 t.Logf("endpoint %s", estr)
204 return server, estr
205}
206
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700207func newCollection(t *testing.T, rootCtx *context.T) (rpc.Server, string) {
Jiri Simsa6ac95222015-02-23 16:11:49 -0800208 server, err := v23.NewServer(rootCtx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700209 if err != nil {
210 boom(t, "r.NewServer: %s", err)
211 }
212 // Start serving on a loopback address.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800213 eps, err := server.Listen(v23.GetListenSpec(rootCtx))
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700214 if err != nil {
215 boom(t, "Failed to Listen mount table: %s", err)
216 }
217 // Add a collection service. This is just a service we can mount
218 // and test against.
219 cPrefix := "collection"
Cosmos Nicolaou92dba582014-11-05 17:24:10 -0800220 if err := server.ServeDispatcher(cPrefix, newCollectionServer()); err != nil {
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700221 boom(t, "Failed to register mock collection: %s", err)
222 }
Cosmos Nicolaou28dabfc2014-12-15 22:51:07 -0800223 estr := eps[0].String()
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700224 t.Logf("endpoint %s", estr)
225 return server, estr
226}
227
228func TestMountTable(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800229 rootCtx, aliceCtx, bobCtx, shutdown := initTest()
230 defer shutdown()
231
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700232 mt, mtAddr := newMT(t, "testdata/test.perms", "", "testMountTable", rootCtx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700233 defer mt.Stop()
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700234 collection, collectionAddr := newCollection(t, rootCtx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700235 defer collection.Stop()
236
237 collectionName := naming.JoinAddressName(collectionAddr, "collection")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700238
239 // Mount the collection server into the mount table.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700240 vlog.Infof("Mount the collection server into the mount table.")
Asim Shankar43d1f932015-03-24 20:57:56 -0700241 doMount(t, rootCtx, mtAddr, "stuff", collectionName, true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700242
243 // Create a few objects and make sure we can read them.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700244 vlog.Infof("Create a few objects.")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800245 export(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/the/rain"), "the rain")
246 export(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/in/spain"), "in spain")
247 export(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/falls"), "falls mainly on the plain")
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700248 vlog.Infof("Make sure we can read them.")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800249 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/the/rain"), "the rain", true)
250 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/in/spain"), "in spain", true)
251 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/falls"), "falls mainly on the plain", true)
252 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "/stuff/falls"), "falls mainly on the plain", true)
253 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/nonexistant"), "falls mainly on the plain", false)
254 checkContents(t, bobCtx, naming.JoinAddressName(mtAddr, "stuff/the/rain"), "the rain", true)
255 checkContents(t, aliceCtx, naming.JoinAddressName(mtAddr, "stuff/the/rain"), "the rain", false)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700256
257 // Test multiple mounts.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700258 vlog.Infof("Multiple mounts.")
Asim Shankar43d1f932015-03-24 20:57:56 -0700259 doMount(t, rootCtx, mtAddr, "a/b", collectionName, true)
260 doMount(t, rootCtx, mtAddr, "x/y", collectionName, true)
261 doMount(t, rootCtx, mtAddr, "alpha//beta", collectionName, true)
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700262 vlog.Infof("Make sure we can read them.")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800263 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuff/falls"), "falls mainly on the plain", true)
264 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "a/b/falls"), "falls mainly on the plain", true)
265 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "x/y/falls"), "falls mainly on the plain", true)
266 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "alpha/beta/falls"), "falls mainly on the plain", true)
267 checkContents(t, aliceCtx, naming.JoinAddressName(mtAddr, "a/b/falls"), "falls mainly on the plain", true)
268 checkContents(t, bobCtx, naming.JoinAddressName(mtAddr, "a/b/falls"), "falls mainly on the plain", false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800269
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700270 // Test getting/setting AccessLists.
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700271 perms, version := doGetPermissions(t, rootCtx, mtAddr, "stuff", true)
272 doSetPermissions(t, rootCtx, mtAddr, "stuff", perms, "xyzzy", false) // bad version
273 doSetPermissions(t, rootCtx, mtAddr, "stuff", perms, version, true) // correct version
Adam Sadovskyb1f9e3c2015-04-08 11:03:49 -0700274 _, nversion := doGetPermissions(t, rootCtx, mtAddr, "stuff", true)
275 if nversion == version {
276 boom(t, "version didn't change after SetPermissions: %s", nversion)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800277 }
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700278 doSetPermissions(t, rootCtx, mtAddr, "stuff", perms, "", true) // no version
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800279
280 // Bob should be able to create nodes under the mounttable root but not alice.
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700281 doSetPermissions(t, aliceCtx, mtAddr, "onlybob", perms, "", false)
282 doSetPermissions(t, bobCtx, mtAddr, "onlybob", perms, "", true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700283
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700284 // Test that setting Permissions to permissions that don't include the the setter's
Suharsh Sivakumar7120abd2015-04-15 13:54:23 -0700285 // blessings in Admin, automatically add their Blessings to Admin to prevent
286 // locking everyone out.
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700287 perms, _ = doGetPermissions(t, bobCtx, mtAddr, "onlybob", true)
288 noRootPerms := perms.Copy()
289 noRootPerms.Clear("bob", "Admin")
290 doSetPermissions(t, bobCtx, mtAddr, "onlybob", noRootPerms, "", true)
Suharsh Sivakumar7120abd2015-04-15 13:54:23 -0700291 // This should succeed, because "bob" should automatically be added to "Admin"
292 // even though he cleared himself from "Admin".
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700293 doSetPermissions(t, bobCtx, mtAddr, "onlybob", perms, "", true)
294 // Test that adding a non-standard perms is normalized when retrieved.
295 admin := perms["Admin"]
Suharsh Sivakumar7120abd2015-04-15 13:54:23 -0700296 admin.In = []security.BlessingPattern{"bob", "bob"}
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700297 perms["Admin"] = admin
298 doSetPermissions(t, bobCtx, mtAddr, "onlybob", perms, "", true)
299 perms, _ = doGetPermissions(t, bobCtx, mtAddr, "onlybob", true)
300 if got, want := perms["Admin"].In, []security.BlessingPattern{"bob"}; !reflect.DeepEqual(got, want) {
Suharsh Sivakumar7120abd2015-04-15 13:54:23 -0700301 boom(t, "got %v, want %v", got, want)
302 }
303
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700304 // Test generic unmount.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700305 vlog.Info("Test generic unmount.")
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800306 doUnmount(t, rootCtx, mtAddr, "a/b", "", true)
307 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "a/b/falls"), "falls mainly on the plain", false)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700308
309 // Test specific unmount.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700310 vlog.Info("Test specific unmount.")
Asim Shankar43d1f932015-03-24 20:57:56 -0700311 doMount(t, rootCtx, mtAddr, "a/b", collectionName, true)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800312 doUnmount(t, rootCtx, mtAddr, "a/b", collectionName, true)
313 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "a/b/falls"), "falls mainly on the plain", false)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700314
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700315 // Try timing out a mount.
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700316 vlog.Info("Try timing out a mount.")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700317 ft := NewFakeTimeClock()
318 setServerListClock(ft)
Asim Shankar43d1f932015-03-24 20:57:56 -0700319 doMount(t, rootCtx, mtAddr, "stuffWithTTL", collectionName, true)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800320 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuffWithTTL/the/rain"), "the rain", true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700321 ft.advance(time.Duration(ttlSecs+4) * time.Second)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800322 checkContents(t, rootCtx, naming.JoinAddressName(mtAddr, "stuffWithTTL/the/rain"), "the rain", false)
Ryan Brownac972652014-05-19 10:57:32 -0700323
David Why Use Two When One Will Do Presotto5add67e2014-07-31 11:14:30 -0700324 // Test unauthorized mount.
325 vlog.Info("Test unauthorized mount.")
Asim Shankar43d1f932015-03-24 20:57:56 -0700326 doMount(t, bobCtx, mtAddr, "/a/b", collectionName, false)
327 doMount(t, aliceCtx, mtAddr, "/a/b", collectionName, false)
Ryan Brownac972652014-05-19 10:57:32 -0700328
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800329 doUnmount(t, bobCtx, mtAddr, "x/y", collectionName, false)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700330}
331
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800332func doGlobX(t *testing.T, ctx *context.T, ep, suffix, pattern string, joinServer bool) []string {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700333 name := naming.JoinAddressName(ep, suffix)
Jiri Simsa6ac95222015-02-23 16:11:49 -0800334 client := v23.GetClient(ctx)
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700335 call, err := client.StartCall(ctx, name, rpc.GlobMethod, []interface{}{pattern}, options.NoResolve{})
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700336 if err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700337 boom(t, "Glob.StartCall %s %s: %s", name, pattern, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700338 }
339 var reply []string
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700340 for {
Todd Wang2331dd02015-03-17 15:38:39 -0700341 var gr naming.GlobReply
David Why Use Two When One Will Do Presottoe5e62d02015-02-19 14:35:20 -0800342 err := call.Recv(&gr)
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700343 if err == io.EOF {
344 break
345 }
346 if err != nil {
347 boom(t, "Glob.StartCall %s: %s", name, pattern, err)
348 }
David Why Use Two When One Will Do Presottoe5e62d02015-02-19 14:35:20 -0800349 switch v := gr.(type) {
Todd Wang2331dd02015-03-17 15:38:39 -0700350 case naming.GlobReplyEntry:
David Why Use Two When One Will Do Presottoe5e62d02015-02-19 14:35:20 -0800351 if joinServer && len(v.Value.Servers) > 0 {
352 reply = append(reply, naming.JoinAddressName(v.Value.Servers[0].Server, v.Value.Name))
353 } else {
354 reply = append(reply, v.Value.Name)
355 }
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700356 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700357 }
Todd Wange77f9952015-02-18 13:20:50 -0800358 if err := call.Finish(); err != nil {
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700359 boom(t, "Glob.Finish %s: %s", name, pattern, err)
Shyam Jayaramanc4aed6e2014-07-22 14:25:06 -0700360 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700361 return reply
362}
363
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800364func doGlob(t *testing.T, ctx *context.T, ep, suffix, pattern string) []string {
365 return doGlobX(t, ctx, ep, suffix, pattern, false)
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700366}
367
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700368// checkMatch verified that the two slices contain the same string items, albeit
369// not necessarily in the same order. Item repetitions are allowed, but their
370// numbers need to match as well.
371func checkMatch(t *testing.T, want []string, got []string) {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800372 if len(want) == 0 && len(got) == 0 {
373 return
374 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700375 w := sort.StringSlice(want)
376 w.Sort()
377 g := sort.StringSlice(got)
378 g.Sort()
379 if !reflect.DeepEqual(w, g) {
380 boom(t, "Glob expected %v got %v", want, got)
381 }
382}
383
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800384// checkExists makes sure a name exists (or not).
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800385func checkExists(t *testing.T, ctx *context.T, ep, suffix string, shouldSucceed bool) {
386 x := doGlobX(t, ctx, ep, "", suffix, false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800387 if len(x) != 1 || x[0] != suffix {
388 if shouldSucceed {
389 boom(t, "Failed to find %s", suffix)
390 }
391 return
392 }
393 if !shouldSucceed {
394 boom(t, "%s exists but shouldn't", suffix)
395 }
396}
397
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700398func TestGlob(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700399 rootCtx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800400 defer shutdown()
401
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700402 server, estr := newMT(t, "", "", "testGlob", rootCtx)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700403 defer server.Stop()
404
405 // set up a mount space
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700406 fakeServer := naming.JoinAddressName(estr, "quux")
Asim Shankar43d1f932015-03-24 20:57:56 -0700407 doMount(t, rootCtx, estr, "one/bright/day", fakeServer, true)
408 doMount(t, rootCtx, estr, "in/the/middle", fakeServer, true)
409 doMount(t, rootCtx, estr, "of/the/night", fakeServer, true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700410
411 // Try various globs.
412 tests := []struct {
413 in string
414 expected []string
415 }{
416 {"*", []string{"one", "in", "of"}},
Bogdan Capritab6b195a2014-06-11 17:55:03 -0700417 {"...", []string{"", "one", "in", "of", "one/bright", "in/the", "of/the", "one/bright/day", "in/the/middle", "of/the/night"}},
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700418 {"*/...", []string{"one", "in", "of", "one/bright", "in/the", "of/the", "one/bright/day", "in/the/middle", "of/the/night"}},
Bogdan Capritab6b195a2014-06-11 17:55:03 -0700419 {"one/...", []string{"one", "one/bright", "one/bright/day"}},
Robin Thellend434e39f2014-08-27 14:46:27 -0700420 {"of/the/night/two/dead/boys", []string{"of/the/night"}},
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700421 {"*/the", []string{"in/the", "of/the"}},
422 {"*/the/...", []string{"in/the", "of/the", "in/the/middle", "of/the/night"}},
423 {"o*", []string{"one", "of"}},
424 {"", []string{""}},
425 }
426 for _, test := range tests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800427 out := doGlob(t, rootCtx, estr, "", test.in)
Ryan Brownac972652014-05-19 10:57:32 -0700428 checkMatch(t, test.expected, out)
429 }
Robin Thellend434e39f2014-08-27 14:46:27 -0700430
431 // Test Glob on a name that is under a mounted server. The result should the
432 // the address the mounted server with the extra suffix.
433 {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800434 results := doGlobX(t, rootCtx, estr, "of/the/night/two/dead/boys/got/up/to/fight", "*", true)
Robin Thellend434e39f2014-08-27 14:46:27 -0700435 if len(results) != 1 {
436 boom(t, "Unexpected number of results. Got %v, want 1", len(results))
437 }
David Why Use Two When One Will Do Presotto9c374c22014-10-21 16:16:30 -0700438 _, suffix := naming.SplitAddressName(results[0])
439 if expected := "quux/two/dead/boys/got/up/to/fight"; suffix != expected {
Robin Thellend434e39f2014-08-27 14:46:27 -0700440 boom(t, "Unexpected suffix. Got %v, want %v", suffix, expected)
441 }
442 }
Ryan Brownac972652014-05-19 10:57:32 -0700443}
444
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700445func TestAccessListTemplate(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800446 rootCtx, aliceCtx, bobCtx, shutdown := initTest()
447 defer shutdown()
448
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700449 server, estr := newMT(t, "testdata/test.perms", "", "testAccessListTemplate", rootCtx)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800450 defer server.Stop()
451 fakeServer := naming.JoinAddressName(estr, "quux")
Tilak Sharmada5165e2014-10-07 16:39:51 -0700452
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800453 // Noone should be able to mount on someone else's names.
Asim Shankar43d1f932015-03-24 20:57:56 -0700454 doMount(t, aliceCtx, estr, "users/ted", fakeServer, false)
455 doMount(t, bobCtx, estr, "users/carol", fakeServer, false)
456 doMount(t, rootCtx, estr, "users/george", fakeServer, false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800457
458 // Anyone should be able to mount on their own names.
Asim Shankar43d1f932015-03-24 20:57:56 -0700459 doMount(t, aliceCtx, estr, "users/alice", fakeServer, true)
460 doMount(t, bobCtx, estr, "users/bob", fakeServer, true)
461 doMount(t, rootCtx, estr, "users/root", fakeServer, true)
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700462
463 // Make sure the counter works.
464 doUnmount(t, aliceCtx, estr, "users/alice", "", true)
465 doUnmount(t, bobCtx, estr, "users/bob", "", true)
466 doUnmount(t, rootCtx, estr, "users/root", "", true)
467 perms := access.Permissions{"Admin": access.AccessList{In: []security.BlessingPattern{security.AllPrincipals}}}
468 doSetPermissions(t, aliceCtx, estr, "users/alice/a/b/c/d", perms, "", true)
469 doSetPermissions(t, aliceCtx, estr, "users/alice/a/b/c/d", perms, "", true)
470
471 // Do we obey limits?
472 for i := 0; i < defaultMaxNodesPerUser-5; i++ {
473 node := fmt.Sprintf("users/alice/a/b/c/d/%d", i)
474 doSetPermissions(t, aliceCtx, estr, node, perms, "", true)
475 }
476 doSetPermissions(t, aliceCtx, estr, "users/alice/a/b/c/d/straw", perms, "", false)
477
478 // See if the stats numbers are correct.
479 testcases := []struct {
480 key string
481 expected interface{}
482 }{
483 {"alice", int64(defaultMaxNodesPerUser)},
484 {"bob", int64(0)},
485 {"root", int64(0)},
486 {localUser, int64(3)},
487 }
488 for _, tc := range testcases {
489 name := "testAccessListTemplate/num-nodes-per-user/" + tc.key
490 got, err := libstats.Value(name)
491 if err != nil {
492 t.Errorf("unexpected error getting map entry for %s: %s", name, err)
493 }
494 if got != tc.expected {
495 t.Errorf("unexpected getting map entry for %s. Got %v, want %v", name, got, tc.expected)
496 }
497 }
498}
499
500func getUserNodeCounts(t *testing.T) (counts map[string]int32) {
501 s, err := libstats.Value("mounttable/num-nodes-per-user")
502 if err != nil {
503 boom(t, "Can't get mounttable statistics")
504 }
505 // This string is a json encoded map. Decode.
506 switch v := s.(type) {
507 default:
508 boom(t, "Wrong type for mounttable statistics")
509 case string:
510 err = json.Unmarshal([]byte(v), &counts)
511 if err != nil {
512 boom(t, "Can't unmarshal mounttable statistics")
513 }
514 }
515 return
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800516}
517
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700518func TestGlobAccessLists(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800519 rootCtx, aliceCtx, bobCtx, shutdown := initTest()
520 defer shutdown()
521
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700522 server, estr := newMT(t, "testdata/test.perms", "", "testGlobAccessLists", rootCtx)
Ryan Brownac972652014-05-19 10:57:32 -0700523 defer server.Stop()
524
525 // set up a mount space
526 fakeServer := naming.JoinAddressName(estr, "quux")
Asim Shankar43d1f932015-03-24 20:57:56 -0700527 doMount(t, aliceCtx, estr, "one/bright/day", fakeServer, false) // Fails because alice can't mount there.
528 doMount(t, bobCtx, estr, "one/bright/day", fakeServer, true)
529 doMount(t, rootCtx, estr, "a/b/c", fakeServer, true)
Ryan Brownac972652014-05-19 10:57:32 -0700530
531 // Try various globs.
532 tests := []struct {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800533 ctx *context.T
Ryan Brownac972652014-05-19 10:57:32 -0700534 in string
535 expected []string
536 }{
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800537 {rootCtx, "*", []string{"one", "a", "stuff", "users"}},
538 {aliceCtx, "*", []string{"one", "a", "users"}},
539 {bobCtx, "*", []string{"one", "stuff", "users"}},
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800540 // bob, alice, and root have different visibility to the space.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800541 {rootCtx, "*/...", []string{"one", "a", "one/bright", "a/b", "one/bright/day", "a/b/c", "stuff", "users"}},
542 {aliceCtx, "*/...", []string{"one", "a", "one/bright", "a/b", "one/bright/day", "a/b/c", "users"}},
543 {bobCtx, "*/...", []string{"one", "one/bright", "one/bright/day", "stuff", "users"}},
Ryan Brownac972652014-05-19 10:57:32 -0700544 }
545 for _, test := range tests {
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800546 out := doGlob(t, test.ctx, estr, "", test.in)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700547 checkMatch(t, test.expected, out)
548 }
549}
550
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800551func TestCleanup(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700552 rootCtx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800553 defer shutdown()
554
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700555 server, estr := newMT(t, "", "", "testCleanup", rootCtx)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800556 defer server.Stop()
557
558 // Set up one mount.
559 fakeServer := naming.JoinAddressName(estr, "quux")
Asim Shankar43d1f932015-03-24 20:57:56 -0700560 doMount(t, rootCtx, estr, "one/bright/day", fakeServer, true)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800561 checkMatch(t, []string{"one", "one/bright", "one/bright/day"}, doGlob(t, rootCtx, estr, "", "*/..."))
562
563 // After the unmount nothing should be left
564 doUnmount(t, rootCtx, estr, "one/bright/day", "", true)
David Why Use Two When One Will Do Presotto4d26a992015-05-06 10:17:09 -0700565 checkMatch(t, nil, doGlob(t, rootCtx, estr, "", "one"))
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800566 checkMatch(t, nil, doGlob(t, rootCtx, estr, "", "*/..."))
567
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700568 // Set up a mount, then set the AccessList.
Asim Shankar43d1f932015-03-24 20:57:56 -0700569 doMount(t, rootCtx, estr, "one/bright/day", fakeServer, true)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800570 checkMatch(t, []string{"one", "one/bright", "one/bright/day"}, doGlob(t, rootCtx, estr, "", "*/..."))
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700571 perms := access.Permissions{"Read": access.AccessList{In: []security.BlessingPattern{security.AllPrincipals}}}
572 doSetPermissions(t, rootCtx, estr, "one/bright", perms, "", true)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800573
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700574 // After the unmount we should still have everything above the AccessList.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800575 doUnmount(t, rootCtx, estr, "one/bright/day", "", true)
576 checkMatch(t, []string{"one", "one/bright"}, doGlob(t, rootCtx, estr, "", "*/..."))
577}
578
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800579func TestDelete(t *testing.T) {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800580 rootCtx, aliceCtx, bobCtx, shutdown := initTest()
581 defer shutdown()
582
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700583 server, estr := newMT(t, "testdata/test.perms", "", "testDelete", rootCtx)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800584 defer server.Stop()
585
586 // set up a mount space
587 fakeServer := naming.JoinAddressName(estr, "quux")
Asim Shankar43d1f932015-03-24 20:57:56 -0700588 doMount(t, bobCtx, estr, "one/bright/day", fakeServer, true)
589 doMount(t, rootCtx, estr, "a/b/c", fakeServer, true)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800590
591 // It shouldn't be possible to delete anything with children unless explicitly requested.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800592 doDeleteNode(t, rootCtx, estr, "a/b", false)
593 checkExists(t, rootCtx, estr, "a/b", true)
594 doDeleteSubtree(t, rootCtx, estr, "a/b", true)
595 checkExists(t, rootCtx, estr, "a/b", false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800596
597 // Alice shouldn't be able to delete what bob created but bob and root should.
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800598 doDeleteNode(t, aliceCtx, estr, "one/bright/day", false)
599 checkExists(t, rootCtx, estr, "one/bright/day", true)
600 doDeleteNode(t, rootCtx, estr, "one/bright/day", true)
601 checkExists(t, rootCtx, estr, "one/bright/day", false)
602 doDeleteNode(t, bobCtx, estr, "one/bright", true)
603 checkExists(t, rootCtx, estr, "one/bright", false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800604}
605
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700606func TestServerFormat(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700607 rootCtx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800608 defer shutdown()
609
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700610 server, estr := newMT(t, "", "", "testerverFormat", rootCtx)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700611 defer server.Stop()
612
Asim Shankar43d1f932015-03-24 20:57:56 -0700613 doMount(t, rootCtx, estr, "endpoint", naming.JoinAddressName(estr, "life/on/the/mississippi"), true)
614 doMount(t, rootCtx, estr, "hostport", "/atrampabroad:8000", true)
615 doMount(t, rootCtx, estr, "invalid/not/rooted", "atrampabroad:8000", false)
616 doMount(t, rootCtx, estr, "invalid/no/port", "/atrampabroad", false)
617 doMount(t, rootCtx, estr, "invalid/endpoint", "/@following the equator:8000@@@", false)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700618}
619
620func TestExpiry(t *testing.T) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700621 rootCtx, shutdown := test.InitForTest()
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800622 defer shutdown()
623
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700624 server, estr := newMT(t, "", "", "testExpiry", rootCtx)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700625 defer server.Stop()
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700626 collection, collectionAddr := newCollection(t, rootCtx)
Cosmos Nicolaoufdc838b2014-06-30 21:44:27 -0700627 defer collection.Stop()
628
629 collectionName := naming.JoinAddressName(collectionAddr, "collection")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700630
631 ft := NewFakeTimeClock()
632 setServerListClock(ft)
Asim Shankar43d1f932015-03-24 20:57:56 -0700633 doMount(t, rootCtx, estr, "a1/b1", collectionName, true)
634 doMount(t, rootCtx, estr, "a1/b2", collectionName, true)
635 doMount(t, rootCtx, estr, "a2/b1", collectionName, true)
636 doMount(t, rootCtx, estr, "a2/b2/c", collectionName, true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700637
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800638 checkMatch(t, []string{"a1/b1", "a2/b1"}, doGlob(t, rootCtx, estr, "", "*/b1/..."))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700639 ft.advance(time.Duration(ttlSecs/2) * time.Second)
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800640 checkMatch(t, []string{"a1/b1", "a2/b1"}, doGlob(t, rootCtx, estr, "", "*/b1/..."))
641 checkMatch(t, []string{"c"}, doGlob(t, rootCtx, estr, "a2/b2", "*"))
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700642 // Refresh only a1/b1. All the other mounts will expire upon the next
643 // ft advance.
Asim Shankar43d1f932015-03-24 20:57:56 -0700644 doMount(t, rootCtx, estr, "a1/b1", collectionName, true)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700645 ft.advance(time.Duration(ttlSecs/2+4) * time.Second)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800646 checkMatch(t, []string{"a1", "a1/b1"}, doGlob(t, rootCtx, estr, "", "*/..."))
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800647 checkMatch(t, []string{"a1/b1"}, doGlob(t, rootCtx, estr, "", "*/b1/..."))
Ryan Brownac972652014-05-19 10:57:32 -0700648}
649
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700650func TestBadAccessLists(t *testing.T) {
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700651 _, err := NewMountTableDispatcher("testdata/invalid.perms", "", "mounttable")
Ryan Brownac972652014-05-19 10:57:32 -0700652 if err == nil {
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700653 boom(t, "Expected json parse error in permissions file")
Ryan Brownac972652014-05-19 10:57:32 -0700654 }
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700655 _, err = NewMountTableDispatcher("testdata/doesntexist.perms", "", "mounttable")
Robin Thellendc4b51692015-02-05 17:11:16 -0800656 if err != nil {
David Why Use Two When One Will Do Presottoba9593f2015-04-22 12:29:26 -0700657 boom(t, "Missing permissions file should not cause an error")
Ryan Brownac972652014-05-19 10:57:32 -0700658 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700659}
Asim Shankar88292912014-10-09 19:41:07 -0700660
Robin Thellend0aed6da2015-04-17 14:39:35 -0700661func getCounter(t *testing.T, ctx *context.T, name string) int64 {
662 st := stats.StatsClient(name)
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700663 v, err := st.Value(ctx)
664 if err != nil {
Robin Thellend0aed6da2015-04-17 14:39:35 -0700665 t.Fatalf("Failed to get %q: %v", name, err)
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700666 return -1
667 }
668 var value int64
669 if err := vdl.Convert(&value, v); err != nil {
Robin Thellend0aed6da2015-04-17 14:39:35 -0700670 t.Fatalf("Unexpected value type for %q: %v", name, err)
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700671 }
672 return value
673}
674
Robin Thellend0aed6da2015-04-17 14:39:35 -0700675func nodeCount(t *testing.T, ctx *context.T, addr string) int64 {
676 name := naming.JoinAddressName(addr, "__debug/stats/mounttable/num-nodes")
677 return getCounter(t, ctx, name)
678}
679
680func serverCount(t *testing.T, ctx *context.T, addr string) int64 {
681 name := naming.JoinAddressName(addr, "__debug/stats/mounttable/num-mounted-servers")
682 return getCounter(t, ctx, name)
683}
684
685func TestStatsCounters(t *testing.T) {
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700686 rootCtx, shutdown := test.InitForTest()
687 defer shutdown()
688
Robin Thellend62909b22015-04-23 13:54:55 -0700689 ft := NewFakeTimeClock()
690 setServerListClock(ft)
691
David Why Use Two When One Will Do Presottoc74ac702015-05-07 15:33:52 -0700692 server, estr := newMT(t, "", "", "mounttable", rootCtx)
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700693 defer server.Stop()
694
695 // Test flat tree
696 for i := 1; i <= 10; i++ {
697 name := fmt.Sprintf("node%d", i)
698 addr := naming.JoinAddressName(estr, name)
699 doMount(t, rootCtx, estr, name, addr, true)
700 if expected, got := int64(i+1), nodeCount(t, rootCtx, estr); got != expected {
701 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
702 }
Robin Thellend0aed6da2015-04-17 14:39:35 -0700703 if expected, got := int64(i), serverCount(t, rootCtx, estr); got != expected {
704 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
705 }
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700706 }
707 for i := 1; i <= 10; i++ {
708 name := fmt.Sprintf("node%d", i)
709 if i%2 == 0 {
710 doUnmount(t, rootCtx, estr, name, "", true)
711 } else {
712 doDeleteSubtree(t, rootCtx, estr, name, true)
713 }
714 if expected, got := int64(11-i), nodeCount(t, rootCtx, estr); got != expected {
715 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
716 }
Robin Thellend0aed6da2015-04-17 14:39:35 -0700717 if expected, got := int64(10-i), serverCount(t, rootCtx, estr); got != expected {
718 t.Errorf("Unexpected number of server. Got %d, expected %d", got, expected)
719 }
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700720 }
721
722 // Test deep tree
723 doMount(t, rootCtx, estr, "1/2/3/4/5/6/7/8/9a/10", naming.JoinAddressName(estr, ""), true)
724 doMount(t, rootCtx, estr, "1/2/3/4/5/6/7/8/9b/11", naming.JoinAddressName(estr, ""), true)
725 if expected, got := int64(13), nodeCount(t, rootCtx, estr); got != expected {
726 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
727 }
Robin Thellend0aed6da2015-04-17 14:39:35 -0700728 if expected, got := int64(2), serverCount(t, rootCtx, estr); got != expected {
729 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
730 }
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700731 doDeleteSubtree(t, rootCtx, estr, "1/2/3/4/5", true)
732 if expected, got := int64(5), nodeCount(t, rootCtx, estr); got != expected {
733 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
734 }
Robin Thellend0aed6da2015-04-17 14:39:35 -0700735 if expected, got := int64(0), serverCount(t, rootCtx, estr); got != expected {
736 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
737 }
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700738 doDeleteSubtree(t, rootCtx, estr, "1", true)
739 if expected, got := int64(1), nodeCount(t, rootCtx, estr); got != expected {
740 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
741 }
Robin Thellend0aed6da2015-04-17 14:39:35 -0700742
743 // Test multiple servers per node
744 for i := 1; i <= 5; i++ {
745 server := naming.JoinAddressName(estr, fmt.Sprintf("addr%d", i))
746 doMount(t, rootCtx, estr, "node1", server, true)
747 doMount(t, rootCtx, estr, "node2", server, true)
748 if expected, got := int64(3), nodeCount(t, rootCtx, estr); got != expected {
749 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
750 }
751 if expected, got := int64(2*i), serverCount(t, rootCtx, estr); got != expected {
752 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
753 }
754 }
755 doUnmount(t, rootCtx, estr, "node1", "", true)
756 if expected, got := int64(2), nodeCount(t, rootCtx, estr); got != expected {
757 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
758 }
759 if expected, got := int64(5), serverCount(t, rootCtx, estr); got != expected {
760 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
761 }
762 for i := 1; i <= 5; i++ {
763 server := naming.JoinAddressName(estr, fmt.Sprintf("addr%d", i))
764 doUnmount(t, rootCtx, estr, "node2", server, true)
765 expectedNodes := int64(2)
766 if i == 5 {
767 expectedNodes = 1
768 }
769 if expected, got := expectedNodes, nodeCount(t, rootCtx, estr); got != expected {
770 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
771 }
772 if expected, got := int64(5-i), serverCount(t, rootCtx, estr); got != expected {
773 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
774 }
775 }
Robin Thellend93b1f6d2015-04-22 12:31:34 -0700776
777 // Mount on an existing intermediate node.
778 doMount(t, rootCtx, estr, "1/2/3/4/5/6/7/8/9a/10", naming.JoinAddressName(estr, ""), true)
779 doMount(t, rootCtx, estr, "1/2/3/4/5", naming.JoinAddressName(estr, ""), true)
780 if expected, got := int64(6), nodeCount(t, rootCtx, estr); got != expected {
781 t.Errorf("Unexpected number of nodes. Got %d, expected %d", got, expected)
782 }
783 if expected, got := int64(1), serverCount(t, rootCtx, estr); got != expected {
784 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
785 }
Robin Thellend62909b22015-04-23 13:54:55 -0700786
787 // Test expired mounts
788 // "1/2/3/4/5" is still mounted from earlier.
789 ft.advance(time.Duration(ttlSecs+4) * time.Second)
790 if _, err := resolve(rootCtx, naming.JoinAddressName(estr, "1/2/3/4/5")); err == nil {
791 t.Errorf("Expected failure. Got success")
792 }
793 if expected, got := int64(0), serverCount(t, rootCtx, estr); got != expected {
794 t.Errorf("Unexpected number of servers. Got %d, expected %d", got, expected)
795 }
Robin Thellendfd82dcc2015-04-14 21:16:59 -0700796}
797
Jiri Simsa6ac95222015-02-23 16:11:49 -0800798func initTest() (rootCtx *context.T, aliceCtx *context.T, bobCtx *context.T, shutdown v23.Shutdown) {
Cosmos Nicolaou1381f8a2015-03-13 09:40:34 -0700799 test.Init()
800 ctx, shutdown := test.InitForTest()
Asim Shankar88292912014-10-09 19:41:07 -0700801 var err error
Todd Wangad492042015-04-17 15:58:40 -0700802 if rootCtx, err = v23.WithPrincipal(ctx, testutil.NewPrincipal("root")); err != nil {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800803 panic("failed to set root principal")
Matt Rosencrantzac32b6c2014-12-01 15:49:18 -0800804 }
Todd Wangad492042015-04-17 15:58:40 -0700805 if aliceCtx, err = v23.WithPrincipal(ctx, testutil.NewPrincipal("alice")); err != nil {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800806 panic("failed to set alice principal")
Asim Shankar88292912014-10-09 19:41:07 -0700807 }
Todd Wangad492042015-04-17 15:58:40 -0700808 if bobCtx, err = v23.WithPrincipal(ctx, testutil.NewPrincipal("bob")); err != nil {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800809 panic("failed to set bob principal")
Asim Shankar88292912014-10-09 19:41:07 -0700810 }
Matt Rosencrantz6edab562015-01-12 11:07:55 -0800811 for _, r := range []*context.T{rootCtx, aliceCtx, bobCtx} {
Asim Shankar8bc78d62015-02-03 16:53:27 -0800812 // A hack to set the namespace roots to a value that won't work.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800813 v23.GetNamespace(r).SetRoots()
Asim Shankar8bc78d62015-02-03 16:53:27 -0800814 // And have all principals recognize each others blessings.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800815 p1 := v23.GetPrincipal(r)
Asim Shankar8bc78d62015-02-03 16:53:27 -0800816 for _, other := range []*context.T{rootCtx, aliceCtx, bobCtx} {
Asim Shankar4a698282015-03-21 21:59:18 -0700817 // testutil.NewPrincipal has already setup each
Asim Shankar8bc78d62015-02-03 16:53:27 -0800818 // principal to use the same blessing for both server
819 // and client activities.
Jiri Simsa6ac95222015-02-23 16:11:49 -0800820 if err := p1.AddToRoots(v23.GetPrincipal(other).BlessingStore().Default()); err != nil {
Asim Shankar88292912014-10-09 19:41:07 -0700821 panic(err)
822 }
823 }
824 }
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800825 return rootCtx, aliceCtx, bobCtx, shutdown
Asim Shankar88292912014-10-09 19:41:07 -0700826}