2013-07-10 11:10:48 +00:00
|
|
|
// +build linux darwin
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2012 Google Inc.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package fs
|
|
|
|
|
|
|
|
import (
|
|
|
|
"log"
|
|
|
|
"os"
|
2013-11-22 01:51:34 +00:00
|
|
|
"strings"
|
2013-07-11 10:17:27 +00:00
|
|
|
"sync"
|
|
|
|
"time"
|
2013-07-10 11:10:48 +00:00
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
"camlistore.org/pkg/blob"
|
2013-07-18 00:31:19 +00:00
|
|
|
"camlistore.org/pkg/schema"
|
2013-07-11 00:56:09 +00:00
|
|
|
"camlistore.org/pkg/search"
|
2013-11-22 01:51:34 +00:00
|
|
|
"camlistore.org/pkg/syncutil"
|
2013-07-10 11:10:48 +00:00
|
|
|
"camlistore.org/third_party/code.google.com/p/rsc/fuse"
|
|
|
|
)
|
|
|
|
|
2013-07-11 10:17:27 +00:00
|
|
|
const refreshTime = 1 * time.Minute
|
|
|
|
|
2013-07-10 11:10:48 +00:00
|
|
|
type rootsDir struct {
|
2013-12-29 08:27:50 +00:00
|
|
|
noXattr
|
2013-07-10 11:10:48 +00:00
|
|
|
fs *CamliFileSystem
|
2013-12-21 22:22:26 +00:00
|
|
|
at time.Time
|
2013-07-11 10:17:27 +00:00
|
|
|
|
|
|
|
mu sync.Mutex // guards following
|
|
|
|
lastQuery time.Time
|
2014-01-06 22:50:17 +00:00
|
|
|
m map[string]blob.Ref // ent name => permanode
|
|
|
|
children map[string]fuse.Node // ent name => child node
|
2013-07-10 11:10:48 +00:00
|
|
|
}
|
|
|
|
|
2013-12-21 22:22:26 +00:00
|
|
|
func (n *rootsDir) isRO() bool {
|
|
|
|
return !n.at.IsZero()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *rootsDir) dirMode() os.FileMode {
|
|
|
|
if n.isRO() {
|
|
|
|
return 0500
|
|
|
|
}
|
|
|
|
return 0700
|
|
|
|
}
|
|
|
|
|
2013-07-10 11:10:48 +00:00
|
|
|
func (n *rootsDir) Attr() fuse.Attr {
|
|
|
|
return fuse.Attr{
|
2013-12-21 22:22:26 +00:00
|
|
|
Mode: os.ModeDir | n.dirMode(),
|
2013-07-10 11:10:48 +00:00
|
|
|
Uid: uint32(os.Getuid()),
|
|
|
|
Gid: uint32(os.Getgid()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *rootsDir) ReadDir(intr fuse.Intr) ([]fuse.Dirent, fuse.Error) {
|
2013-07-11 10:17:27 +00:00
|
|
|
n.mu.Lock()
|
|
|
|
defer n.mu.Unlock()
|
|
|
|
if err := n.condRefresh(); err != nil {
|
2013-07-10 11:10:48 +00:00
|
|
|
return nil, fuse.EIO
|
|
|
|
}
|
|
|
|
var ents []fuse.Dirent
|
2013-07-11 10:17:27 +00:00
|
|
|
for name := range n.m {
|
|
|
|
ents = append(ents, fuse.Dirent{Name: name})
|
2013-07-10 11:10:48 +00:00
|
|
|
}
|
2013-12-29 08:27:50 +00:00
|
|
|
log.Printf("rootsDir.ReadDir() -> %v", ents)
|
2013-07-10 11:10:48 +00:00
|
|
|
return ents, nil
|
|
|
|
}
|
|
|
|
|
2013-12-19 07:08:10 +00:00
|
|
|
func (n *rootsDir) Remove(req *fuse.RemoveRequest, intr fuse.Intr) fuse.Error {
|
2013-12-21 22:22:26 +00:00
|
|
|
if n.isRO() {
|
|
|
|
return fuse.EPERM
|
|
|
|
}
|
2013-12-19 07:08:10 +00:00
|
|
|
n.mu.Lock()
|
|
|
|
defer n.mu.Unlock()
|
|
|
|
|
|
|
|
if err := n.condRefresh(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
br := n.m[req.Name]
|
|
|
|
if !br.Valid() {
|
|
|
|
return fuse.ENOENT
|
|
|
|
}
|
|
|
|
|
|
|
|
claim := schema.NewDelAttributeClaim(br, "camliRoot", "")
|
|
|
|
_, err := n.fs.client.UploadAndSignBlob(claim)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("rootsDir.Remove:", err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(n.m, req.Name)
|
2014-01-06 22:50:17 +00:00
|
|
|
delete(n.children, req.Name)
|
2013-12-19 07:08:10 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-12-23 08:31:46 +00:00
|
|
|
func (n *rootsDir) Rename(req *fuse.RenameRequest, newDir fuse.Node, intr fuse.Intr) fuse.Error {
|
|
|
|
log.Printf("rootsDir.Rename %q -> %q", req.OldName, req.NewName)
|
2013-12-21 22:22:26 +00:00
|
|
|
if n.isRO() {
|
|
|
|
return fuse.EPERM
|
|
|
|
}
|
|
|
|
|
2013-12-23 08:31:46 +00:00
|
|
|
n.mu.Lock()
|
|
|
|
target, exists := n.m[req.OldName]
|
|
|
|
_, collision := n.m[req.NewName]
|
|
|
|
n.mu.Unlock()
|
|
|
|
if !exists {
|
|
|
|
log.Printf("*rootsDir.Rename src name %q isn't known", req.OldName)
|
|
|
|
return fuse.ENOENT
|
|
|
|
}
|
|
|
|
if collision {
|
|
|
|
log.Printf("*rootsDir.Rename dest %q already exists", req.NewName)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't allow renames if the root contains content. Rename
|
|
|
|
// is mostly implemented to make GUIs that create directories
|
|
|
|
// before asking for the directory name.
|
|
|
|
res, err := n.fs.client.Describe(&search.DescribeRequest{BlobRef: target})
|
|
|
|
if err != nil {
|
|
|
|
log.Println("rootsDir.Rename:", err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
db := res.Meta[target.String()]
|
|
|
|
if db == nil {
|
|
|
|
log.Printf("Failed to pull meta for target: %v", target)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
for k := range db.Permanode.Attr {
|
|
|
|
const p = "camliPath:"
|
|
|
|
if strings.HasPrefix(k, p) {
|
|
|
|
log.Printf("Found file in %q: %q, disallowing rename", req.OldName, k[len(p):])
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
claim := schema.NewSetAttributeClaim(target, "camliRoot", req.NewName)
|
|
|
|
_, err = n.fs.client.UploadAndSignBlob(claim)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Upload rename link error: %v", err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
// Comment transplanted from mutDir.Rename
|
|
|
|
// TODO(bradfitz): this locking would be racy, if the kernel
|
|
|
|
// doesn't do it properly. (It should) Let's just trust the
|
|
|
|
// kernel for now. Later we can verify and remove this
|
|
|
|
// comment.
|
|
|
|
n.mu.Lock()
|
|
|
|
if n.m[req.OldName] != target {
|
|
|
|
panic("Race.")
|
|
|
|
}
|
|
|
|
delete(n.m, req.OldName)
|
2014-01-06 22:50:17 +00:00
|
|
|
delete(n.children, req.OldName)
|
|
|
|
delete(n.children, req.NewName)
|
2013-12-23 08:31:46 +00:00
|
|
|
n.m[req.NewName] = target
|
|
|
|
n.mu.Unlock()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-10 11:10:48 +00:00
|
|
|
func (n *rootsDir) Lookup(name string, intr fuse.Intr) (fuse.Node, fuse.Error) {
|
2013-07-11 00:56:09 +00:00
|
|
|
log.Printf("fs.roots: Lookup(%q)", name)
|
2013-07-11 10:17:27 +00:00
|
|
|
n.mu.Lock()
|
|
|
|
defer n.mu.Unlock()
|
|
|
|
if err := n.condRefresh(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
br := n.m[name]
|
2013-08-04 02:54:30 +00:00
|
|
|
if !br.Valid() {
|
2013-07-10 11:10:48 +00:00
|
|
|
return nil, fuse.ENOENT
|
|
|
|
}
|
2014-01-06 22:50:17 +00:00
|
|
|
|
|
|
|
nod, ok := n.children[name]
|
|
|
|
if ok {
|
|
|
|
return nod, nil
|
|
|
|
}
|
|
|
|
|
2013-12-21 22:22:26 +00:00
|
|
|
if n.isRO() {
|
|
|
|
nod = newRODir(n.fs, br, name, n.at)
|
|
|
|
} else {
|
|
|
|
nod = &mutDir{
|
|
|
|
fs: n.fs,
|
|
|
|
permanode: br,
|
|
|
|
name: name,
|
2014-01-18 02:15:13 +00:00
|
|
|
xattrs: map[string][]byte{},
|
2013-12-21 22:22:26 +00:00
|
|
|
}
|
2013-07-10 11:10:48 +00:00
|
|
|
}
|
2014-01-06 22:50:17 +00:00
|
|
|
n.children[name] = nod
|
|
|
|
|
2013-07-10 11:10:48 +00:00
|
|
|
return nod, nil
|
|
|
|
}
|
2013-07-11 10:17:27 +00:00
|
|
|
|
|
|
|
// requires n.mu is held
|
|
|
|
func (n *rootsDir) condRefresh() fuse.Error {
|
|
|
|
if n.lastQuery.After(time.Now().Add(-refreshTime)) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
log.Printf("fs.roots: querying")
|
|
|
|
|
2013-11-22 01:51:34 +00:00
|
|
|
var rootRes, impRes *search.WithAttrResponse
|
|
|
|
var grp syncutil.Group
|
|
|
|
grp.Go(func() (err error) {
|
|
|
|
rootRes, err = n.fs.client.GetPermanodesWithAttr(&search.WithAttrRequest{N: 100, Attr: "camliRoot"})
|
|
|
|
return
|
|
|
|
})
|
|
|
|
grp.Go(func() (err error) {
|
|
|
|
impRes, err = n.fs.client.GetPermanodesWithAttr(&search.WithAttrRequest{N: 100, Attr: "camliImportRoot"})
|
|
|
|
return
|
|
|
|
})
|
|
|
|
if err := grp.Err(); err != nil {
|
2013-07-11 10:17:27 +00:00
|
|
|
log.Printf("fs.recent: GetRecentPermanodes error in ReadDir: %v", err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
2014-01-11 04:50:20 +00:00
|
|
|
n.m = make(map[string]blob.Ref)
|
|
|
|
if n.children == nil {
|
|
|
|
n.children = make(map[string]fuse.Node)
|
|
|
|
}
|
|
|
|
|
2013-07-11 10:17:27 +00:00
|
|
|
dr := &search.DescribeRequest{
|
|
|
|
Depth: 1,
|
|
|
|
}
|
2013-11-22 01:51:34 +00:00
|
|
|
for _, wi := range rootRes.WithAttr {
|
|
|
|
dr.BlobRefs = append(dr.BlobRefs, wi.Permanode)
|
|
|
|
}
|
|
|
|
for _, wi := range impRes.WithAttr {
|
2013-07-11 10:17:27 +00:00
|
|
|
dr.BlobRefs = append(dr.BlobRefs, wi.Permanode)
|
|
|
|
}
|
2014-01-11 04:50:20 +00:00
|
|
|
if len(dr.BlobRefs) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-11 10:17:27 +00:00
|
|
|
dres, err := n.fs.client.Describe(dr)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Describe failure: %v", err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
|
2013-11-22 01:51:34 +00:00
|
|
|
// Roots
|
2014-01-07 22:46:08 +00:00
|
|
|
currentRoots := map[string]bool{}
|
2013-11-22 01:51:34 +00:00
|
|
|
for _, wi := range rootRes.WithAttr {
|
2013-07-11 10:17:27 +00:00
|
|
|
pn := wi.Permanode
|
|
|
|
db := dres.Meta[pn.String()]
|
|
|
|
if db != nil && db.Permanode != nil {
|
|
|
|
name := db.Permanode.Attr.Get("camliRoot")
|
|
|
|
if name != "" {
|
2014-01-07 22:46:08 +00:00
|
|
|
currentRoots[name] = true
|
2013-07-11 10:17:27 +00:00
|
|
|
n.m[name] = pn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-11-22 01:51:34 +00:00
|
|
|
|
2014-01-07 22:46:08 +00:00
|
|
|
// Remove any children objects we have mapped that are no
|
|
|
|
// longer relevant.
|
|
|
|
for name := range n.children {
|
|
|
|
if !currentRoots[name] {
|
|
|
|
delete(n.children, name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-22 01:51:34 +00:00
|
|
|
// Importers (mapped as roots for now)
|
|
|
|
for _, wi := range impRes.WithAttr {
|
|
|
|
pn := wi.Permanode
|
|
|
|
db := dres.Meta[pn.String()]
|
|
|
|
if db != nil && db.Permanode != nil {
|
|
|
|
name := db.Permanode.Attr.Get("camliImportRoot")
|
|
|
|
if name != "" {
|
|
|
|
name = strings.Replace(name, ":", "-", -1)
|
|
|
|
name = strings.Replace(name, "/", "-", -1)
|
|
|
|
n.m["importer-"+name] = pn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-11 10:17:27 +00:00
|
|
|
n.lastQuery = time.Now()
|
|
|
|
return nil
|
|
|
|
}
|
2013-07-18 00:31:19 +00:00
|
|
|
|
|
|
|
func (n *rootsDir) Mkdir(req *fuse.MkdirRequest, intr fuse.Intr) (fuse.Node, fuse.Error) {
|
2013-12-21 22:22:26 +00:00
|
|
|
if n.isRO() {
|
|
|
|
return nil, fuse.EPERM
|
|
|
|
}
|
|
|
|
|
2013-07-18 00:31:19 +00:00
|
|
|
name := req.Name
|
|
|
|
|
|
|
|
// Create a Permanode for the root.
|
|
|
|
pr, err := n.fs.client.UploadNewPermanode()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("rootsDir.Create(%q): %v", name, err)
|
|
|
|
return nil, fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a camliRoot attribute to the root permanode.
|
|
|
|
claim := schema.NewSetAttributeClaim(pr.BlobRef, "camliRoot", name)
|
|
|
|
_, err = n.fs.client.UploadAndSignBlob(claim)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("rootsDir.Create(%q): %v", name, err)
|
|
|
|
return nil, fuse.EIO
|
|
|
|
}
|
|
|
|
|
|
|
|
nod := &mutDir{
|
|
|
|
fs: n.fs,
|
|
|
|
permanode: pr.BlobRef,
|
|
|
|
name: name,
|
2013-12-29 08:27:50 +00:00
|
|
|
xattrs: map[string][]byte{},
|
2013-07-18 00:31:19 +00:00
|
|
|
}
|
|
|
|
n.mu.Lock()
|
|
|
|
n.m[name] = pr.BlobRef
|
|
|
|
n.mu.Unlock()
|
|
|
|
|
|
|
|
return nod, nil
|
|
|
|
}
|