1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: Apache-2.0
13 // lookupnode is a caching tree node that is initially empty and calls
14 // loadOne and loadAll to load/update child nodes as needed.
16 // See (*customFileSystem)MountUsers for example usage.
17 type lookupnode struct {
19 loadOne func(parent inode, name string) (inode, error)
20 loadAll func(parent inode) ([]inode, error)
21 stale func(time.Time) bool
26 staleOne map[string]time.Time
29 // Sync flushes pending writes for loaded children and, if successful,
30 // triggers a reload on next lookup.
31 func (ln *lookupnode) Sync() error {
32 err := ln.treenode.Sync()
37 ln.staleAll = time.Time{}
43 func (ln *lookupnode) Readdir() ([]os.FileInfo, error) {
45 defer ln.staleLock.Unlock()
46 checkTime := time.Now()
47 if ln.stale(ln.staleAll) {
48 all, err := ln.loadAll(ln)
52 for _, child := range all {
54 _, err = ln.treenode.Child(child.FileInfo().Name(), func(inode) (inode, error) {
62 ln.staleAll = checkTime
63 // No value in ln.staleOne can make a difference to an
64 // "entry is stale?" test now, because no value is
65 // newer than ln.staleAll. Reclaim memory.
68 return ln.treenode.Readdir()
71 // Child rejects (with ErrInvalidOperation) calls to add/replace
72 // children, instead calling loadOne when a non-existing child is
74 func (ln *lookupnode) Child(name string, replace func(inode) (inode, error)) (inode, error) {
76 defer ln.staleLock.Unlock()
77 checkTime := time.Now()
80 if ln.stale(ln.staleAll) && ln.stale(ln.staleOne[name]) {
81 existing, err = ln.treenode.Child(name, func(inode) (inode, error) {
82 return ln.loadOne(ln, name)
84 if err == nil && existing != nil {
85 if ln.staleOne == nil {
86 ln.staleOne = map[string]time.Time{name: checkTime}
88 ln.staleOne[name] = checkTime
92 existing, err = ln.treenode.Child(name, nil)
93 if err != nil && !os.IsNotExist(err) {
98 // Let the callback try to delete or replace the
99 // existing node; if it does, return
100 // ErrInvalidOperation.
101 if tryRepl, err := replace(existing); err != nil {
102 // Propagate error from callback
104 } else if tryRepl != existing {
105 return existing, ErrInvalidOperation
108 // Return original error from ln.treenode.Child() (it might be