2013-07-05 17:41:07 +00:00
|
|
|
// +build linux darwin
|
|
|
|
|
2013-02-07 05:57:07 +00:00
|
|
|
/*
|
|
|
|
Copyright 2013 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 (
|
2013-02-08 05:55:17 +00:00
|
|
|
"log"
|
2013-02-08 06:43:50 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"sync"
|
2013-02-18 19:02:26 +00:00
|
|
|
"time"
|
2013-02-07 05:57:07 +00:00
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
"camlistore.org/pkg/blob"
|
2013-02-08 05:55:17 +00:00
|
|
|
"camlistore.org/pkg/search"
|
2013-02-07 05:57:07 +00:00
|
|
|
|
|
|
|
"camlistore.org/third_party/code.google.com/p/rsc/fuse"
|
|
|
|
)
|
|
|
|
|
|
|
|
// recentDir implements fuse.Node and is a directory of recent
|
|
|
|
// permanodes' files, for permanodes with a camliContent pointing to a
|
|
|
|
// "file".
|
|
|
|
type recentDir struct {
|
|
|
|
fs *CamliFileSystem
|
2013-02-08 06:43:50 +00:00
|
|
|
|
2013-02-18 19:02:26 +00:00
|
|
|
mu sync.Mutex
|
|
|
|
ents map[string]*search.DescribedBlob // filename to blob meta
|
|
|
|
modTime map[string]time.Time // filename to permanode modtime
|
2013-02-07 05:57:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (n *recentDir) Attr() fuse.Attr {
|
|
|
|
return fuse.Attr{
|
|
|
|
Mode: os.ModeDir | 0700,
|
|
|
|
Uid: uint32(os.Getuid()),
|
|
|
|
Gid: uint32(os.Getgid()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *recentDir) ReadDir(intr fuse.Intr) ([]fuse.Dirent, fuse.Error) {
|
2013-02-08 06:43:50 +00:00
|
|
|
log.Printf("fs.recent: ReadDir / searching")
|
|
|
|
n.mu.Lock()
|
|
|
|
defer n.mu.Unlock()
|
|
|
|
|
|
|
|
n.ents = make(map[string]*search.DescribedBlob)
|
2013-02-18 19:02:26 +00:00
|
|
|
n.modTime = make(map[string]time.Time)
|
2013-02-08 06:43:50 +00:00
|
|
|
|
2013-02-08 05:55:17 +00:00
|
|
|
req := &search.RecentRequest{N: 100}
|
|
|
|
res, err := n.fs.client.GetRecentPermanodes(req)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("fs.recent: GetRecentPermanodes error in ReadDir: %v", err)
|
|
|
|
return nil, fuse.EIO
|
|
|
|
}
|
|
|
|
|
2013-02-07 05:57:07 +00:00
|
|
|
var ents []fuse.Dirent
|
2013-02-08 06:43:50 +00:00
|
|
|
for _, ri := range res.Recent {
|
2013-02-19 05:32:46 +00:00
|
|
|
modTime := ri.ModTime.Time()
|
2013-02-08 06:43:50 +00:00
|
|
|
meta := res.Meta.Get(ri.BlobRef)
|
|
|
|
if meta == nil || meta.Permanode == nil {
|
|
|
|
continue
|
|
|
|
}
|
2013-08-04 02:54:30 +00:00
|
|
|
cc, ok := blob.Parse(meta.Permanode.Attr.Get("camliContent"))
|
|
|
|
if !ok {
|
2013-02-08 06:43:50 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
ccMeta := res.Meta.Get(cc)
|
2013-02-17 19:43:03 +00:00
|
|
|
if ccMeta == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
var name string
|
|
|
|
switch {
|
|
|
|
case ccMeta.File != nil:
|
|
|
|
name = ccMeta.File.FileName
|
2013-02-19 05:32:46 +00:00
|
|
|
if mt := ccMeta.File.Time; !mt.IsZero() {
|
|
|
|
modTime = mt.Time()
|
|
|
|
}
|
2013-02-17 19:43:03 +00:00
|
|
|
case ccMeta.Dir != nil:
|
|
|
|
name = ccMeta.Dir.FileName
|
|
|
|
default:
|
2013-02-08 06:43:50 +00:00
|
|
|
continue
|
|
|
|
}
|
2013-02-17 07:04:18 +00:00
|
|
|
if name == "" || n.ents[name] != nil {
|
2013-02-08 06:43:50 +00:00
|
|
|
name = ccMeta.BlobRef.String() + path.Ext(name)
|
|
|
|
if n.ents[name] != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
n.ents[name] = ccMeta
|
2013-02-19 05:32:46 +00:00
|
|
|
n.modTime[name] = modTime
|
|
|
|
log.Printf("fs.recent: name %q = %v (at %v -> %v)", name, ccMeta.BlobRef, ri.ModTime.Time(), modTime)
|
2013-02-08 06:43:50 +00:00
|
|
|
ents = append(ents, fuse.Dirent{
|
|
|
|
Name: name,
|
|
|
|
})
|
|
|
|
}
|
2013-02-17 07:04:18 +00:00
|
|
|
log.Printf("fs.recent returning %d entries", len(ents))
|
2013-02-07 05:57:07 +00:00
|
|
|
return ents, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *recentDir) Lookup(name string, intr fuse.Intr) (fuse.Node, fuse.Error) {
|
2013-02-08 06:43:50 +00:00
|
|
|
n.mu.Lock()
|
|
|
|
defer n.mu.Unlock()
|
2013-02-17 19:43:03 +00:00
|
|
|
if n.ents == nil {
|
|
|
|
// Odd case: a Lookup before a Readdir. Force a readdir to
|
|
|
|
// seed our map. Mostly hit just during development.
|
2013-02-18 19:02:26 +00:00
|
|
|
n.mu.Unlock() // release, since ReadDir will acquire
|
2013-02-17 19:43:03 +00:00
|
|
|
n.ReadDir(intr)
|
|
|
|
n.mu.Lock()
|
|
|
|
}
|
2013-02-08 06:43:50 +00:00
|
|
|
db := n.ents[name]
|
|
|
|
log.Printf("fs.recent: Lookup(%q) = %v", name, db)
|
|
|
|
if db == nil {
|
|
|
|
return nil, fuse.ENOENT
|
2013-02-07 16:05:51 +00:00
|
|
|
}
|
2013-02-18 19:02:26 +00:00
|
|
|
nod := &node{
|
|
|
|
fs: n.fs,
|
|
|
|
blobref: db.BlobRef,
|
|
|
|
pnodeModTime: n.modTime[name],
|
|
|
|
}
|
|
|
|
return nod, nil
|
2013-02-07 05:57:07 +00:00
|
|
|
}
|