blob: f4e059fd73739cd079785e507d4b04a4272a724a [file] [log] [blame]
Jiri Simsad7616c92015-03-24 23:44:30 -07001// Copyright 2015 The Vanadium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Jiri Simsa5293dcb2014-05-10 09:56:38 -07005package mounttable
6
7import (
Ryan Brownac972652014-05-19 10:57:32 -07008 "encoding/json"
Ryan Brownac972652014-05-19 10:57:32 -07009 "os"
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080010 "reflect"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070011 "strings"
12 "sync"
13 "time"
14
Jiri Simsaffceefa2015-02-28 11:03:34 -080015 "v.io/x/ref/lib/glob"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070016
Jiri Simsa6ac95222015-02-23 16:11:49 -080017 "v.io/v23"
Matt Rosencrantz250558f2015-03-17 11:37:31 -070018 "v.io/v23/context"
Jiri Simsa6ac95222015-02-23 16:11:49 -080019 "v.io/v23/naming"
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070020 "v.io/v23/rpc"
Jiri Simsa6ac95222015-02-23 16:11:49 -080021 "v.io/v23/security"
Todd Wang387d8a42015-03-30 17:09:05 -070022 "v.io/v23/security/access"
Jiri Simsa6ac95222015-02-23 16:11:49 -080023 "v.io/v23/services/mounttable"
Jiri Simsa6ac95222015-02-23 16:11:49 -080024 "v.io/v23/verror"
Jiri Simsa337af232015-02-27 14:36:46 -080025 "v.io/x/lib/vlog"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070026)
27
Mike Burrowscf766812015-03-31 16:01:50 -070028const pkgPath = "v.io/x/ref/services/mounttable/lib"
29
Jiri Simsa5293dcb2014-05-10 09:56:38 -070030var (
Mike Burrowscf766812015-03-31 16:01:50 -070031 errMalformedAddress = verror.Register(pkgPath+".errMalformedAddress", verror.NoRetry, "{1:}{2:} malformed address {3} for mounted server {4}{:_}")
32 errMTDoesntMatch = verror.Register(pkgPath+".errMTDoesntMatch", verror.NoRetry, "{1:}{2:} MT doesn't match{:_}")
33 errLeafDoesntMatch = verror.Register(pkgPath+".errLeafDoesntMatch", verror.NoRetry, "{1:}{2:} Leaf doesn't match{:_}")
34 errCantDeleteRoot = verror.Register(pkgPath+".errCantDeleteRoot", verror.NoRetry, "{1:}{2:} cannot delete root node{:_}")
35 errNotEmpty = verror.Register(pkgPath+".errNotEmpty", verror.NoRetry, "{1:}{2:} cannot delete {3}: has children{:_}")
36 errNamingLoop = verror.Register(pkgPath+".errNamingLoop", verror.NoRetry, "{1:}{2:} Loop in namespace{:_}")
37)
38
39var (
40 traverseTags = []mounttable.Tag{mounttable.Read, mounttable.Resolve, mounttable.Create, mounttable.Admin}
41 createTags = []mounttable.Tag{mounttable.Create, mounttable.Admin}
42 removeTags = []mounttable.Tag{mounttable.Admin}
43 mountTags = []mounttable.Tag{mounttable.Mount, mounttable.Admin}
44 resolveTags = []mounttable.Tag{mounttable.Read, mounttable.Resolve, mounttable.Admin}
45 globTags = []mounttable.Tag{mounttable.Read, mounttable.Admin}
46 setTags = []mounttable.Tag{mounttable.Admin}
47 getTags = []mounttable.Tag{mounttable.Admin, mounttable.Read}
48 allTags = []mounttable.Tag{mounttable.Read, mounttable.Resolve, mounttable.Admin, mounttable.Mount, mounttable.Create}
Jiri Simsa5293dcb2014-05-10 09:56:38 -070049)
50
51// mountTable represents a namespace. One exists per server instance.
52type mountTable struct {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080053 root *node
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070054 superUsers access.AccessList
Jiri Simsa5293dcb2014-05-10 09:56:38 -070055}
56
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070057var _ rpc.Dispatcher = (*mountTable)(nil)
Benjamin Prosnitzfdfbf7b2014-10-08 09:47:21 -070058
David Why Use Two When One Will Do Presottocfea09a2014-05-12 10:45:40 -070059// mountContext represents a client bind. The name is the name that was bound to.
Jiri Simsa5293dcb2014-05-10 09:56:38 -070060type mountContext struct {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080061 name string
62 elems []string // parsed elements of name
63 mt *mountTable
Jiri Simsa5293dcb2014-05-10 09:56:38 -070064}
65
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080066// mount represents a single mount point. It contains the rooted names of all servers mounted
Jiri Simsa5293dcb2014-05-10 09:56:38 -070067// here. The servers are considered equivalent, i.e., RPCs to a name below this
68// point can be sent to any of these servers.
69type mount struct {
70 servers *serverList
David Why Use Two When One Will Do Presotto3da1c792014-10-03 11:15:53 -070071 mt bool
Robin Thellend89e95232015-03-24 13:48:48 -070072 leaf bool
Jiri Simsa5293dcb2014-05-10 09:56:38 -070073}
74
Jiri Simsa5293dcb2014-05-10 09:56:38 -070075// node is a single point in the tree representing the mount table.
76type node struct {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -080077 sync.RWMutex
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070078 parent *node
79 mount *mount
80 children map[string]*node
81 acls *TAMG
82 amTemplate access.Permissions
83 explicitAccessLists bool
Jiri Simsa5293dcb2014-05-10 09:56:38 -070084}
85
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080086const templateVar = "%%"
87
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070088// NewMountTableDispatcher creates a new server that uses the AccessLists specified in
Asim Shankar68885192014-11-26 12:48:35 -080089// aclfile for authorization.
90//
91// aclfile is a JSON-encoded mapping from paths in the mounttable to the
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070092// access.Permissions for that path. The tags used in the map are the typical
Todd Wang387d8a42015-03-30 17:09:05 -070093// access tags (the Tag type defined in v.io/v23/security/access).
Matt Rosencrantz94502cf2015-03-18 09:43:44 -070094func NewMountTableDispatcher(aclfile string) (rpc.Dispatcher, error) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -080095 mt := &mountTable{
96 root: new(node),
97 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -080098 mt.root.parent = new(node) // just for its lock
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -070099 if err := mt.parseAccessLists(aclfile); err != nil && !os.IsNotExist(err) {
Ryan Brownac972652014-05-19 10:57:32 -0700100 return nil, err
101 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800102 return mt, nil
Ryan Brownac972652014-05-19 10:57:32 -0700103}
104
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700105func (mt *mountTable) parseAccessLists(path string) error {
106 vlog.VI(2).Infof("parseAccessLists(%s)", path)
Ryan Brownac972652014-05-19 10:57:32 -0700107 if path == "" {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800108 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700109 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700110 var tams map[string]access.Permissions
Ryan Brownac972652014-05-19 10:57:32 -0700111 f, err := os.Open(path)
112 if err != nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800113 return err
Ryan Brownac972652014-05-19 10:57:32 -0700114 }
115 defer f.Close()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800116 if err = json.NewDecoder(f).Decode(&tams); err != nil {
117 return err
Ryan Brownac972652014-05-19 10:57:32 -0700118 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800119 for name, tam := range tams {
120 var elems []string
121 isPattern := false
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700122 // Create name and add the AccessList map to it.
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800123 if len(name) == 0 {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700124 // If the config file has is an Admin tag on the root AccessList, the
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800125 // list of Admin users is the equivalent of a super user for
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700126 // the whole table. This is not updated if the AccessList is later
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800127 // modified.
128 if acl, exists := tam[string(mounttable.Admin)]; exists {
129 mt.superUsers = acl
130 }
131 } else {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700132 // AccessList templates terminate with a %% element. These are very
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800133 // constrained matches, i.e., the trailing element of the name
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700134 // is copied into every %% in the AccessList.
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800135 elems = strings.Split(name, "/")
136 if elems[len(elems)-1] == templateVar {
137 isPattern = true
138 elems = elems[:len(elems)-1]
139 }
140 }
141
142 n, err := mt.findNode(nil, elems, true, nil)
143 if n != nil || err == nil {
144 vlog.VI(2).Infof("added tam %v to %s", tam, name)
145 if isPattern {
146 n.amTemplate = tam
147 } else {
148 n.acls, _ = n.acls.Set("", tam)
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700149 n.explicitAccessLists = true
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800150 }
Asim Shankar68885192014-11-26 12:48:35 -0800151 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800152 n.parent.Unlock()
153 n.Unlock()
Ryan Brownac972652014-05-19 10:57:32 -0700154 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800155 return nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700156}
157
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700158// Lookup implements rpc.Dispatcher.Lookup.
Robin Thellenda02fe8f2014-11-19 09:58:29 -0800159func (mt *mountTable) Lookup(name string) (interface{}, security.Authorizer, error) {
Cosmos Nicolaou69335402014-05-20 14:41:58 -0700160 vlog.VI(2).Infof("*********************Lookup %s", name)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700161 ms := &mountContext{
162 name: name,
163 mt: mt,
164 }
165 if len(name) > 0 {
166 ms.elems = strings.Split(name, "/")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700167 }
Cosmos Nicolaou710daa22014-11-11 19:39:18 -0800168 return mounttable.MountTableServer(ms), ms, nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700169}
170
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700171// isActive returns true if a mount has unexpired servers attached.
172func (m *mount) isActive() bool {
173 if m == nil {
174 return false
175 }
176 return m.servers.removeExpired() > 0
177}
178
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800179// satisfies returns no error if the ctx + n.acls satisfies the associated one of the required Tags.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700180func (n *node) satisfies(mt *mountTable, call rpc.ServerCall, tags []mounttable.Tag) error {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700181 // No AccessLists means everything (for now).
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800182 if call == nil || tags == nil || n.acls == nil {
Ryan Brownac972652014-05-19 10:57:32 -0700183 return nil
184 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800185 // "Self-RPCs" are always authorized.
Matt Rosencrantz311378b2015-03-25 15:26:12 -0700186 secCall := security.GetCall(call.Context())
187 if l, r := secCall.LocalBlessings().PublicKey(), secCall.RemoteBlessings().PublicKey(); l != nil && reflect.DeepEqual(l, r) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800188 return nil
Ryan Brownac972652014-05-19 10:57:32 -0700189 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700190 // Match client's blessings against the AccessLists.
Ankur9e75e7f2015-03-18 18:48:41 -0700191 blessings, invalidB := security.RemoteBlessingNames(call.Context())
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800192 for _, tag := range tags {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700193 if acl, exists := n.acls.GetPermissionsForTag(string(tag)); exists && acl.Includes(blessings...) {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800194 return nil
195 }
196 }
197 if mt.superUsers.Includes(blessings...) {
198 return nil
199 }
Ryan Brown41093a92015-02-10 10:59:14 -0800200 if len(invalidB) > 0 {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800201 return verror.New(verror.ErrNoAccess, call.Context(), blessings, invalidB)
Ryan Brown41093a92015-02-10 10:59:14 -0800202 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800203 return verror.New(verror.ErrNoAccess, call.Context(), blessings)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800204}
205
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700206func expand(acl *access.AccessList, name string) *access.AccessList {
207 newAccessList := new(access.AccessList)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800208 for _, bp := range acl.In {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700209 newAccessList.In = append(newAccessList.In, security.BlessingPattern(strings.Replace(string(bp), templateVar, name, -1)))
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800210 }
211 for _, bp := range acl.NotIn {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700212 newAccessList.NotIn = append(newAccessList.NotIn, strings.Replace(bp, templateVar, name, -1))
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800213 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700214 return newAccessList
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800215}
216
217// satisfiesTemplate returns no error if the ctx + n.amTemplate satisfies the associated one of
218// the required Tags.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700219func (n *node) satisfiesTemplate(call rpc.ServerCall, tags []mounttable.Tag, name string) error {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800220 if n.amTemplate == nil {
221 return nil
222 }
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700223 // Match client's blessings against the AccessLists.
Ankur9e75e7f2015-03-18 18:48:41 -0700224 blessings, invalidB := security.RemoteBlessingNames(call.Context())
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800225 for _, tag := range tags {
226 if acl, exists := n.amTemplate[string(tag)]; exists && expand(&acl, name).Includes(blessings...) {
227 return nil
228 }
229 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800230 return verror.New(verror.ErrNoAccess, call.Context(), blessings, invalidB)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800231}
232
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700233// copyAccessLists copies one nodes AccessLists to another and adds the clients blessings as
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800234// patterns to the Admin tag.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700235func copyAccessLists(call rpc.ServerCall, cur *node) *TAMG {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800236 if call == nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800237 return nil
238 }
239 if cur.acls == nil {
240 return nil
241 }
242 acls := cur.acls.Copy()
Ankur9e75e7f2015-03-18 18:48:41 -0700243 blessings, _ := security.RemoteBlessingNames(call.Context())
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800244 for _, b := range blessings {
245 acls.Add(security.BlessingPattern(b), string(mounttable.Admin))
246 }
247 return acls
248}
249
250// createTAMGFromTemplate creates a new TAMG from the template subsituting name for %% everywhere.
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700251func createTAMGFromTemplate(tam access.Permissions, name string) *TAMG {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800252 tamg := NewTAMG()
253 for tag, acl := range tam {
254 tamg.tam[tag] = *expand(&acl, name)
255 }
256 return tamg
257}
258
259// traverse returns the node for the path represented by elems. If none exists and create is false, return nil.
260// Otherwise create the path and return a pointer to the terminal node. If a mount point is encountered
261// while following the path, return that node and any remaining elems.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800262//
263// If it returns a node, both the node and its parent are locked.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700264func (mt *mountTable) traverse(call rpc.ServerCall, elems []string, create bool) (*node, []string, error) {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800265 // Invariant is that the current node and its parent are both locked.
266 cur := mt.root
267 cur.parent.Lock()
268 cur.Lock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800269 for i, e := range elems {
270 vlog.VI(2).Infof("satisfying %v %v", elems[0:i], *cur)
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800271 if call != nil {
272 if err := cur.satisfies(mt, call, traverseTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800273 cur.parent.Unlock()
274 cur.Unlock()
275 return nil, nil, err
276 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800277 }
278 // If we hit another mount table, we're done.
279 if cur.mount.isActive() {
280 return cur, elems[i:], nil
281 }
282 // Walk the children looking for a match.
283 c, ok := cur.children[e]
284 if ok {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800285 cur.parent.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800286 cur = c
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800287 cur.Lock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800288 continue
289 }
290 if !create {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800291 cur.parent.Unlock()
292 cur.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800293 return nil, nil, nil
294 }
295 // Create a new node and keep recursing.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800296 cur.parent.Unlock()
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800297 if err := cur.satisfies(mt, call, createTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800298 cur.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800299 return nil, nil, err
300 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800301 if call != nil {
302 if err := cur.satisfiesTemplate(call, createTags, e); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800303 cur.Unlock()
304 return nil, nil, err
305 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800306 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800307 // At this point cur is still locked, OK to use and change it.
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800308 next := new(node)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800309 next.parent = cur
310 if cur.amTemplate != nil {
311 next.acls = createTAMGFromTemplate(cur.amTemplate, e)
312 } else {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700313 next.acls = copyAccessLists(call, cur)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800314 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800315 if cur.children == nil {
316 cur.children = make(map[string]*node)
317 }
318 cur.children[e] = next
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800319 cur = next
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800320 cur.Lock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800321 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800322 // Only way out of the loop is via a return or exhausting all elements. In
323 // the latter case both cur and cur.parent are locked.
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800324 return cur, nil, nil
325}
326
327// findNode finds a node in the table and optionally creates a path to it.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800328//
329// If a node is found, on return it and its parent are locked.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700330func (mt *mountTable) findNode(call rpc.ServerCall, elems []string, create bool, tags []mounttable.Tag) (*node, error) {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800331 n, nelems, err := mt.traverse(call, elems, create)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800332 if err != nil {
333 return nil, err
334 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800335 if n == nil {
336 return nil, nil
337 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800338 if len(nelems) > 0 {
339 n.parent.Unlock()
340 n.Unlock()
341 return nil, nil
342 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800343 if err := n.satisfies(mt, call, tags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800344 n.parent.Unlock()
345 n.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800346 return nil, err
347 }
348 return n, nil
349}
350
351// findMountPoint returns the first mount point encountered in the path and
352// any elements remaining of the path.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800353//
354// If a mountpoint is found, on return it and its parent are locked.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700355func (mt *mountTable) findMountPoint(call rpc.ServerCall, elems []string) (*node, []string, error) {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800356 n, nelems, err := mt.traverse(call, elems, false)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800357 if err != nil {
358 return nil, nil, err
359 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800360 if n == nil {
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800361 return nil, nil, nil
362 }
David Why Use Two When One Will Do Presottoc67ad7c2015-03-04 14:27:42 -0800363 // If we can't resolve it, we can't use it.
364 if err := n.satisfies(mt, call, resolveTags); err != nil {
365 n.parent.Unlock()
366 n.Unlock()
367 return nil, nil, err
368 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800369 if !n.mount.isActive() {
370 removed := n.removeUseless()
371 n.parent.Unlock()
372 n.Unlock()
373 // If we removed the node, see if we can remove any of its
374 // ascendants.
375 if removed {
376 mt.removeUselessRecursive(elems[:len(elems)-1])
377 }
378 return nil, nil, nil
379 }
380 return n, nelems, nil
Ryan Brownac972652014-05-19 10:57:32 -0700381}
382
Ryan Brownac972652014-05-19 10:57:32 -0700383// Authorize verifies that the client has access to the requested node.
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800384// Since we do the check at the time of access, we always return OK here.
Matt Rosencrantz250558f2015-03-17 11:37:31 -0700385func (ms *mountContext) Authorize(*context.T) error {
Ryan Brownac972652014-05-19 10:57:32 -0700386 return nil
387}
388
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700389// ResolveStep returns the next server in a resolution, the name remaining below that server,
390// and whether or not that server is another mount table.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700391func (ms *mountContext) ResolveStepX(call rpc.ServerCall) (entry naming.MountEntry, err error) {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800392 return ms.ResolveStep(call)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700393}
394
David Why Use Two When One Will Do Presottod3aa6632015-01-20 10:15:39 -0800395// ResolveStep returns the next server in a resolution in the form of a MountEntry. The name
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700396// in the mount entry is the name relative to the server's root.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700397func (ms *mountContext) ResolveStep(call rpc.ServerCall) (entry naming.MountEntry, err error) {
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700398 vlog.VI(2).Infof("ResolveStep %q", ms.name)
399 mt := ms.mt
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700400 // Find the next mount point for the name.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800401 n, elems, werr := mt.findMountPoint(call, ms.elems)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800402 if werr != nil {
403 err = werr
404 return
405 }
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700406 if n == nil {
407 entry.Name = ms.name
408 if len(ms.elems) == 0 {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800409 err = verror.New(naming.ErrNoSuchNameRoot, call.Context(), ms.name)
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700410 } else {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800411 err = verror.New(naming.ErrNoSuchName, call.Context(), ms.name)
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700412 }
413 return
414 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800415 n.parent.Unlock()
416 defer n.Unlock()
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700417 entry.Servers = n.mount.servers.copyToSlice()
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800418 entry.Name = strings.Join(elems, "/")
Todd Wang2331dd02015-03-17 15:38:39 -0700419 entry.ServesMountTable = n.mount.mt
Robin Thellend89e95232015-03-24 13:48:48 -0700420 entry.IsLeaf = n.mount.leaf
David Why Use Two When One Will Do Presotto6f9f5742014-10-20 16:27:05 -0700421 return
422}
423
Todd Wang1aa57692014-11-11 13:53:29 -0800424func hasMTFlag(flags naming.MountFlag) bool {
425 return (flags & naming.MT) == naming.MT
David Why Use Two When One Will Do Presotto3da1c792014-10-03 11:15:53 -0700426}
427
Robin Thellend89e95232015-03-24 13:48:48 -0700428func hasLeafFlag(flags naming.MountFlag) bool {
429 return (flags & naming.Leaf) == naming.Leaf
430}
431
Todd Wang1aa57692014-11-11 13:53:29 -0800432func hasReplaceFlag(flags naming.MountFlag) bool {
433 return (flags & naming.Replace) == naming.Replace
David Why Use Two When One Will Do Presotto3da1c792014-10-03 11:15:53 -0700434}
435
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700436// Mount a server onto the name in the receiver.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700437func (ms *mountContext) Mount(call rpc.ServerCall, server string, ttlsecs uint32, flags naming.MountFlag) error {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700438 mt := ms.mt
439 if ttlsecs == 0 {
440 ttlsecs = 10 * 365 * 24 * 60 * 60 // a really long time
441 }
Asim Shankar43d1f932015-03-24 20:57:56 -0700442 vlog.VI(2).Infof("*********************Mount %q -> %s", ms.name, server)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700443
Cosmos Nicolaou8bd8e102015-01-13 21:52:53 -0800444 // Make sure the server address is reasonable.
445 epString := server
446 if naming.Rooted(server) {
447 epString, _ = naming.SplitAddressName(server)
448 }
Jiri Simsa6ac95222015-02-23 16:11:49 -0800449 _, err := v23.NewEndpoint(epString)
David Why Use Two When One Will Do Presotto59a254c2014-10-30 13:09:29 -0700450 if err != nil {
Mike Burrowscf766812015-03-31 16:01:50 -0700451 return verror.New(errMalformedAddress, call.Context(), epString, server)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700452 }
453
454 // Find/create node in namespace and add the mount.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800455 n, werr := mt.findNode(call, ms.elems, true, mountTags)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800456 if werr != nil {
457 return werr
458 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700459 if n == nil {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800460 return verror.New(naming.ErrNoSuchNameRoot, call.Context(), ms.name)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700461 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800462 // We don't need the parent lock
463 n.parent.Unlock()
464 defer n.Unlock()
David Why Use Two When One Will Do Presotto3da1c792014-10-03 11:15:53 -0700465 if hasReplaceFlag(flags) {
466 n.mount = nil
467 }
David Why Use Two When One Will Do Presottoc28686e2014-11-05 11:19:29 -0800468 wantMT := hasMTFlag(flags)
Robin Thellend89e95232015-03-24 13:48:48 -0700469 wantLeaf := hasLeafFlag(flags)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700470 if n.mount == nil {
Robin Thellend89e95232015-03-24 13:48:48 -0700471 n.mount = &mount{servers: newServerList(), mt: wantMT, leaf: wantLeaf}
David Why Use Two When One Will Do Presotto3da1c792014-10-03 11:15:53 -0700472 } else {
David Why Use Two When One Will Do Presotto59a254c2014-10-30 13:09:29 -0700473 if wantMT != n.mount.mt {
Mike Burrowscf766812015-03-31 16:01:50 -0700474 return verror.New(errMTDoesntMatch, call.Context())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700475 }
Robin Thellend89e95232015-03-24 13:48:48 -0700476 if wantLeaf != n.mount.leaf {
Mike Burrowscf766812015-03-31 16:01:50 -0700477 return verror.New(errLeafDoesntMatch, call.Context())
Robin Thellend89e95232015-03-24 13:48:48 -0700478 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700479 }
Asim Shankar43d1f932015-03-24 20:57:56 -0700480 n.mount.servers.add(server, time.Duration(ttlsecs)*time.Second)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700481 return nil
482}
483
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800484// fullName is for debugging only and should not normally be called.
485func (n *node) fullName() string {
486 if n.parent == nil || n.parent.parent == nil {
487 return ""
488 }
489 for k, c := range n.parent.children {
490 if c == n {
491 return n.parent.fullName() + "/" + k
492 }
493 }
494 return n.parent.fullName() + "/" + "?"
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700495}
496
497// removeUseless removes a node and all of its ascendants that are not useful.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800498//
499// We assume both n and n.parent are locked.
500func (n *node) removeUseless() bool {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700501 if len(n.children) > 0 || n.mount.isActive() || n.explicitAccessLists {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800502 return false
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700503 }
504 for k, c := range n.parent.children {
505 if c == n {
506 delete(n.parent.children, k)
507 break
508 }
509 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800510 return true
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700511}
512
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800513// removeUselessRecursive removes any useless nodes on the tail of the path.
514func (mt *mountTable) removeUselessRecursive(elems []string) {
515 for i := len(elems); i > 0; i-- {
516 n, nelems, _ := mt.traverse(nil, elems[:i-1], false)
517 if n == nil {
518 break
519 }
520 if nelems != nil {
521 n.parent.Unlock()
522 n.Unlock()
523 break
524 }
525 removed := n.removeUseless()
526 n.parent.Unlock()
527 n.Unlock()
528 if !removed {
529 break
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700530 }
531 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700532}
533
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700534// Unmount removes servers from the name in the receiver. If server is specified, only that
535// server is removed.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700536func (ms *mountContext) Unmount(call rpc.ServerCall, server string) error {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800537 vlog.VI(2).Infof("*********************Unmount %q, %s", ms.name, server)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700538 mt := ms.mt
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800539 n, err := mt.findNode(call, ms.elems, false, mountTags)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800540 if err != nil {
541 return err
542 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700543 if n == nil {
544 return nil
545 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700546 if server == "" {
547 n.mount = nil
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800548 } else if n.mount != nil && n.mount.servers.remove(server) == 0 {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700549 n.mount = nil
550 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800551 removed := n.removeUseless()
552 n.parent.Unlock()
553 n.Unlock()
554 if removed {
555 // If we removed the node, see if we can also remove
556 // any of its ascendants.
557 mt.removeUselessRecursive(ms.elems[:len(ms.elems)-1])
558 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700559 return nil
560}
561
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800562// Delete removes the receiver. If all is true, any subtree is also removed.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700563func (ms *mountContext) Delete(call rpc.ServerCall, deleteSubTree bool) error {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800564 vlog.VI(2).Infof("*********************Delete %q, %v", ms.name, deleteSubTree)
565 if len(ms.elems) == 0 {
566 // We can't delete the root.
Mike Burrowscf766812015-03-31 16:01:50 -0700567 return verror.New(errCantDeleteRoot, call.Context())
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800568 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800569 mt := ms.mt
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800570 // Find and lock the parent node.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800571 n, err := mt.findNode(call, ms.elems, false, removeTags)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800572 if err != nil {
573 return err
574 }
575 if n == nil {
576 return nil
577 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800578 defer n.parent.Unlock()
579 defer n.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800580 if !deleteSubTree && len(n.children) > 0 {
Mike Burrowscf766812015-03-31 16:01:50 -0700581 return verror.New(errNotEmpty, call.Context(), ms.name)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800582 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800583 delete(n.parent.children, ms.elems[len(ms.elems)-1])
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800584 return nil
585}
586
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700587// A struct holding a partial result of Glob.
588type globEntry struct {
589 n *node
590 name string
591}
592
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800593// globStep is called with n and n.parent locked. Returns with both unlocked.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700594func (mt *mountTable) globStep(n *node, name string, pattern *glob.Glob, call rpc.ServerCall, ch chan<- naming.GlobReply) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700595 vlog.VI(2).Infof("globStep(%s, %s)", name, pattern)
596
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700597 // If this is a mount point, we're done.
598 if m := n.mount; m != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800599 removed := n.removeUseless()
600 if removed {
601 n.parent.Unlock()
602 n.Unlock()
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700603 return
604 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800605 // Don't need the parent lock anymore.
606 n.parent.Unlock()
Todd Wang2331dd02015-03-17 15:38:39 -0700607 me := naming.MountEntry{
David Why Use Two When One Will Do Presottoc67ad7c2015-03-04 14:27:42 -0800608 Name: name,
609 }
610 // Only fill in the mount info if we can resolve this name.
611 if err := n.satisfies(mt, call, resolveTags); err == nil {
612 me.Servers = m.servers.copyToSlice()
Todd Wang2331dd02015-03-17 15:38:39 -0700613 me.ServesMountTable = n.mount.mt
Robin Thellend00643742015-04-01 10:36:50 -0700614 me.IsLeaf = n.mount.leaf
David Why Use Two When One Will Do Presottoc67ad7c2015-03-04 14:27:42 -0800615 } else {
Todd Wang2331dd02015-03-17 15:38:39 -0700616 me.Servers = []naming.MountedServer{}
Robin Thellend39ac3232014-12-02 09:50:41 -0800617 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800618 // Unlock while we are sending on the channel to avoid livelock.
619 n.Unlock()
Todd Wang2331dd02015-03-17 15:38:39 -0700620 ch <- naming.GlobReplyEntry{me}
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700621 return
622 }
623
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800624 if !pattern.Finished() {
625 // Recurse through the children. OK if client has read access to the
626 // directory or has traverse access to the directory and any access to the child.
627 allAllowed := true
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800628 if err := n.satisfies(mt, call, globTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800629 allAllowed = false
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800630 if err := n.satisfies(mt, call, traverseTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800631 n.parent.Unlock()
632 n.Unlock()
633 return
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800634 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700635 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800636 children := make(map[string]*node, len(n.children))
637 for k, c := range n.children {
638 children[k] = c
639 }
640 n.parent.Unlock()
641 for k, c := range children {
642 // At this point, n lock is held.
643 if ok, _, suffix := pattern.MatchInitialSegment(k); ok {
644 c.Lock()
645 if !allAllowed {
646 // If child allows any access show it. Otherwise, skip.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800647 if err := c.satisfies(mt, call, allTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800648 c.Unlock()
649 continue
650 }
651 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800652 mt.globStep(c, naming.Join(name, k), suffix, call, ch)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800653 n.Lock()
654 }
655 }
656 // Relock the node and its parent in the correct order.
657 // Safe to access n.parent when its unlocked because it never changes.
658 n.Unlock()
659 n.parent.Lock()
660 n.Lock()
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700661 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800662
663 // Remove if no longer useful.
664 if n.removeUseless() || pattern.Len() != 0 {
665 n.parent.Unlock()
666 n.Unlock()
667 return
668 }
669
670 // To see anything, one has to have some access to the node. Don't need the parent lock anymore.
671 n.parent.Unlock()
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800672 if err := n.satisfies(mt, call, allTags); err != nil {
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800673 n.Unlock()
674 return
675 }
676 // Unlock while we are sending on the channel to avoid livelock.
677 n.Unlock()
David Why Use Two When One Will Do Presotto06966782015-03-18 17:00:25 -0700678 // Intermediate nodes are marked as serving a mounttable since they answer the mounttable methods.
679 ch <- naming.GlobReplyEntry{naming.MountEntry{Name: name, ServesMountTable: true}}
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700680}
681
682// Glob finds matches in the namespace. If we reach a mount point before matching the
683// whole pattern, return that mount point.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800684//
Suharsh Sivakumar8646ba62015-03-18 15:22:28 -0700685// pattern is a glob pattern as defined by the v.io/x/ref/lib/glob package.
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800686//
687// To avoid livelocking an application, Glob grabs and releases locks as it descends the tree
688// and holds no locks while writing to the channel. As such a glob can interleave with other
689// operations that add or remove nodes. The result returned by glob may, therefore, represent
690// a state that never existed in the mounttable. For example, if someone removes c/d and later
691// adds a/b while a Glob is in progress, the Glob may return a set of nodes that includes both
692// c/d and a/b.
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700693func (ms *mountContext) Glob__(call rpc.ServerCall, pattern string) (<-chan naming.GlobReply, error) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700694 vlog.VI(2).Infof("mt.Glob %v", ms.elems)
695
696 g, err := glob.Parse(pattern)
697 if err != nil {
Robin Thellend39ac3232014-12-02 09:50:41 -0800698 return nil, err
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700699 }
700
701 mt := ms.mt
Todd Wang2331dd02015-03-17 15:38:39 -0700702 ch := make(chan naming.GlobReply)
Robin Thellend39ac3232014-12-02 09:50:41 -0800703 go func() {
704 defer close(ch)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800705 // If there was an access error, just ignore the entry, i.e., make it invisible.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800706 n, err := mt.findNode(call, ms.elems, false, nil)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800707 if err != nil {
708 return
709 }
Robin Thellend39ac3232014-12-02 09:50:41 -0800710 // If the current name is not fully resolvable on this nameserver we
711 // don't need to evaluate the glob expression. Send a partially resolved
712 // name back to the client.
Robin Thellend39ac3232014-12-02 09:50:41 -0800713 if n == nil {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800714 ms.linkToLeaf(call, ch)
Robin Thellend39ac3232014-12-02 09:50:41 -0800715 return
716 }
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800717 mt.globStep(n, "", g, call, ch)
Robin Thellend39ac3232014-12-02 09:50:41 -0800718 }()
719 return ch, nil
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700720}
Robin Thellend434e39f2014-08-27 14:46:27 -0700721
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700722func (ms *mountContext) linkToLeaf(call rpc.ServerCall, ch chan<- naming.GlobReply) {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800723 n, elems, err := ms.mt.findMountPoint(call, ms.elems)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800724 if err != nil || n == nil {
Robin Thellend434e39f2014-08-27 14:46:27 -0700725 return
726 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800727 n.parent.Unlock()
Robin Thellend434e39f2014-08-27 14:46:27 -0700728 servers := n.mount.servers.copyToSlice()
729 for i, s := range servers {
David Why Use Two When One Will Do Presottoadf0ca12014-11-13 10:49:01 -0800730 servers[i].Server = naming.Join(s.Server, strings.Join(elems, "/"))
Robin Thellend434e39f2014-08-27 14:46:27 -0700731 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800732 n.Unlock()
Todd Wang2331dd02015-03-17 15:38:39 -0700733 ch <- naming.GlobReplyEntry{naming.MountEntry{Name: "", Servers: servers}}
Robin Thellend434e39f2014-08-27 14:46:27 -0700734}
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800735
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700736func (ms *mountContext) SetPermissions(call rpc.ServerCall, tam access.Permissions, etag string) error {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700737 vlog.VI(2).Infof("SetPermissions %q", ms.name)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800738 mt := ms.mt
739
740 // Find/create node in namespace and add the mount.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800741 n, err := mt.findNode(call, ms.elems, true, setTags)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800742 if err != nil {
743 return err
744 }
745 if n == nil {
746 // TODO(p): can this even happen?
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800747 return verror.New(naming.ErrNoSuchName, call.Context(), ms.name)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800748 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800749 n.parent.Unlock()
750 defer n.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800751 n.acls, err = n.acls.Set(etag, tam)
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800752 if err == nil {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700753 n.explicitAccessLists = true
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800754 }
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800755 return err
756}
757
Matt Rosencrantz94502cf2015-03-18 09:43:44 -0700758func (ms *mountContext) GetPermissions(call rpc.ServerCall) (access.Permissions, string, error) {
Benjamin Prosnitzb60efb92015-03-11 17:47:43 -0700759 vlog.VI(2).Infof("GetPermissions %q", ms.name)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800760 mt := ms.mt
761
762 // Find node in namespace and add the mount.
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800763 n, err := mt.findNode(call, ms.elems, false, getTags)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800764 if err != nil {
765 return nil, "", err
766 }
767 if n == nil {
Matt Rosencrantz9dce9b22015-03-02 10:48:37 -0800768 return nil, "", verror.New(naming.ErrNoSuchName, call.Context(), ms.name)
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800769 }
David Why Use Two When One Will Do Presottoa4100172015-01-14 09:25:45 -0800770 n.parent.Unlock()
771 defer n.Unlock()
David Why Use Two When One Will Do Presottoab2bcf22015-01-05 13:14:01 -0800772 etag, tam := n.acls.Get()
773 return tam, etag, nil
774}