Merge branch 'master' of camlistore.org:camlistore

This commit is contained in:
Brett Slatkin 2011-03-22 22:50:35 -07:00
commit f6a9c55cd8
4 changed files with 336 additions and 16 deletions

View File

@ -280,7 +280,7 @@ sub find_go_camli_deps {
my $t = $targets{$target} or die "Bogus or undeclared build target: $target\n"; my $t = $targets{$target} or die "Bogus or undeclared build target: $target\n";
opendir(my $dh, $target) or die; opendir(my $dh, $target) or die;
my @go_files = grep { !/_testmain\.go$/ } grep { /\.go$/ } readdir($dh); my @go_files = grep { !m!^\.\#! } grep { !/_testmain\.go$/ } grep { /\.go$/ } readdir($dh);
closedir($dh); closedir($dh);
# TODO: just stat the files first and keep a cache file of the # TODO: just stat the files first and keep a cache file of the
@ -292,7 +292,7 @@ sub find_go_camli_deps {
my @deps; my @deps;
my %seen; # $dep -> 1 my %seen; # $dep -> 1
for my $f (@go_files) { for my $f (@go_files) {
open(my $fh, "$target/$f") or die; open(my $fh, "$target/$f") or die "Failed to open $target/$f: $!";
my $src = do { local $/; <$fh>; }; my $src = do { local $/; <$fh>; };
unless ($src =~ m!\bimport\s*\((.+?)\)!s) { unless ($src =~ m!\bimport\s*\((.+?)\)!s) {
die "Failed to parse imports from $target/$f.\n". die "Failed to parse imports from $target/$f.\n".
@ -327,7 +327,7 @@ sub gen_target_makefile {
my @deps = @{$t->{deps}}; my @deps = @{$t->{deps}};
opendir(my $dh, $target) or die; opendir(my $dh, $target) or die;
my @go_files = grep { !/_testmain\.go$/ } grep { /\.go$/ } readdir($dh); my @go_files = grep { !m!^\.\#! } grep { !/_testmain\.go$/ } grep { /\.go$/ } readdir($dh);
closedir($dh); closedir($dh);
open(my $mf, ">$target/Makefile") or die; open(my $mf, ">$target/Makefile") or die;

300
clients/go/cammount/fs.go Normal file
View File

@ -0,0 +1,300 @@
/*
Copyright 2011 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 main
import (
"fmt"
"log"
"json"
"os"
"path/filepath"
"strconv"
"sync"
"syscall"
"camli/blobref"
"camli/client"
"camli/schema"
"camli/third_party/github.com/hanwen/go-fuse/fuse"
)
var _ = fmt.Println
var _ = log.Println
type CamliFileSystem struct {
fetcher blobref.Fetcher
root *blobref.BlobRef
lk sync.Mutex
nameToBlob map[string]*blobref.BlobRef
}
func NewCamliFileSystem(client *client.Client, root *blobref.BlobRef) *CamliFileSystem {
return &CamliFileSystem{
fetcher: client,
root: root,
nameToBlob: make(map[string]*blobref.BlobRef),
}
}
// Where name == "" for root,
// Returns nil on failure
func (fs *CamliFileSystem) blobRefFromNameCached(name string) *blobref.BlobRef {
fs.lk.Lock()
defer fs.lk.Unlock()
return fs.nameToBlob[name]
}
func (fs *CamliFileSystem) fetchSchemaSuperset(br *blobref.BlobRef) (*schema.Superset, os.Error) {
rsc, _, err := fs.fetcher.Fetch(br)
if err != nil {
return nil, err
}
defer rsc.Close()
jd := json.NewDecoder(rsc)
ss := new(schema.Superset)
err = jd.Decode(ss)
if err != nil {
log.Printf("Error parsing %s as schema blob: %v", br, err)
return nil, os.EINVAL
}
return ss, nil
}
// Where name == "" for root,
// Returns fuse.Status == fuse.OK on success or anything else on failure.
func (fs *CamliFileSystem) blobRefFromName(name string) (*blobref.BlobRef, fuse.Status) {
if name == "" {
return fs.root, fuse.OK
}
if br := fs.blobRefFromNameCached(name); br != nil {
return br, fuse.OK
}
dir, fileName := filepath.Split(name)
dirBlob, fuseStatus := fs.blobRefFromName(dir)
if fuseStatus != fuse.OK {
return nil, fuseStatus
}
dirss, err := fs.fetchSchemaSuperset(dirBlob)
switch {
case err == os.ENOENT:
log.Printf("Failed to find directory %s", dirBlob)
return nil, fuse.ENOENT
case err == os.EINVAL:
log.Printf("Failed to parse directory %s", dirBlob)
return nil, fuse.ENOTDIR
case err != nil:
panic(fmt.Sprintf("Invalid fetcher error: %v", err))
case dirss == nil:
panic("nil dirss")
case dirss.Type != "directory":
log.Printf("Expected %s to be a directory; actually a %s",
dirBlob, dirss.Type)
return nil, fuse.ENOTDIR
}
if dirss.Entries == "" {
log.Printf("Expected %s to have 'entries'", dirBlob)
return nil, fuse.ENOTDIR
}
entriesBlob := blobref.Parse(dirss.Entries)
if entriesBlob == nil {
log.Printf("Blob %s had invalid blobref %q for its 'entries'", dirBlob, dirss.Entries)
return nil, fuse.ENOTDIR
}
entss, err := fs.fetchSchemaSuperset(entriesBlob)
switch {
case err == os.ENOENT:
log.Printf("Failed to find entries %s via directory %s", entriesBlob, dirBlob)
return nil, fuse.ENOENT
case err == os.EINVAL:
log.Printf("Failed to parse entries %s via directory %s", entriesBlob, dirBlob)
return nil, fuse.ENOTDIR
case err != nil:
panic(fmt.Sprintf("Invalid fetcher error: %v", err))
case entss == nil:
panic("nil entss")
case entss.Type != "static-set":
log.Printf("Expected %s to be a directory; actually a %s",
dirBlob, dirss.Type)
return nil, fuse.ENOTDIR
}
wg := new(sync.WaitGroup)
foundCh := make(chan *blobref.BlobRef)
for _, m := range entss.Members {
wg.Add(1)
go func(memberBlobstr string) {
childss, err := fs.fetchSchemaSuperset(entriesBlob)
if err == nil && childss.HasFilename(fileName) {
foundCh <- entriesBlob
}
wg.Done()
}(m)
}
failCh := make(chan string)
go func() {
wg.Wait()
failCh <- "ENOENT"
}()
select {
case found := <-foundCh:
fs.lk.Lock()
defer fs.lk.Unlock()
fs.nameToBlob[name] = found
return found, fuse.OK
case <-failCh:
}
// TODO: negative cache
return nil, fuse.ENOENT
}
func (fs *CamliFileSystem) Mount(connector *fuse.PathFileSystemConnector) fuse.Status {
log.Printf("cammount: Mount")
return fuse.OK
}
func (fs *CamliFileSystem) Unmount() {
log.Printf("cammount: Unmount.")
}
func (fs *CamliFileSystem) GetAttr(name string) (*fuse.Attr, fuse.Status) {
log.Printf("cammount: GetAttr(%q)", name)
blobref, errStatus := fs.blobRefFromName(name)
log.Printf("cammount: GetAttr(%q), blobRefFromName err=%v", name, errStatus)
if errStatus != fuse.OK {
return nil, errStatus
}
log.Printf("cammount: got blob %s", blobref)
// TODO: this is redundant with what blobRefFromName already
// did. we should at least keep this in RAM (pre-de-JSON'd)
// so we don't have to fetch + unmarshal it again.
ss, err := fs.fetchSchemaSuperset(blobref)
if err != nil {
log.Printf("cammount: GetAttr(%q, %s): fetch schema error: %v", name, blobref, err)
return nil, fuse.EIO
}
out := new(fuse.Attr)
var fi os.FileInfo
if ss.UnixPermission != "" {
// Convert from octal
mode, err := strconv.Btoui64(ss.UnixPermission, 8)
if err != nil {
fi.Mode = uint32(mode)
}
}
// TODO: have a mode to set permissions equal to mounting user?
fi.Uid = ss.UnixOwnerId
fi.Gid = ss.UnixGroupId
// TODO: other types
switch ss.Type {
case "directory":
fi.Mode = fi.Mode | syscall.S_IFDIR
case "file":
fi.Mode = fi.Mode | syscall.S_IFREG
fi.Size = ss.Size
case "symlink":
fi.Mode = fi.Mode | syscall.S_IFLNK
}
// TODO: mtime and such
fuse.CopyFileInfo(&fi, out)
return out, fuse.OK
}
func (fs *CamliFileSystem) Access(name string, mode uint32) fuse.Status {
log.Printf("cammount: Access(%q, %d)", name, mode)
return fuse.OK
}
func (fs *CamliFileSystem) Open(name string, flags uint32) (file fuse.RawFuseFile, code fuse.Status) {
log.Printf("cammount: Open(%q, %d)", name, flags)
// TODO
return nil, fuse.EACCES
}
func (fs *CamliFileSystem) OpenDir(name string) (stream chan fuse.DirEntry, code fuse.Status) {
log.Printf("cammount: OpenDir(%q)", name)
// TODO
return nil, fuse.EACCES
}
func (fs *CamliFileSystem) Readlink(name string) (string, fuse.Status) {
log.Printf("cammount: Readlink(%q)", name)
// TODO
return "", fuse.EACCES
}
// *************************************************************************
// EACCESS stuff
func (fs *CamliFileSystem) Chmod(name string, mode uint32) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Chown(name string, uid uint32, gid uint32) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Create(name string, flags uint32, mode uint32) (file fuse.RawFuseFile, code fuse.Status) {
code = fuse.EACCES
return
}
func (fs *CamliFileSystem) Link(oldName string, newName string) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Mkdir(name string, mode uint32) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Mknod(name string, mode uint32, dev uint32) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Rename(oldName string, newName string) (code fuse.Status) {
return fuse.EACCES
}
func (fs *CamliFileSystem) Rmdir(name string) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Symlink(value string, linkName string) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Truncate(name string, offset uint64) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Unlink(name string) fuse.Status {
return fuse.EACCES
}
func (fs *CamliFileSystem) Utimens(name string, AtimeNs uint64, CtimeNs uint64) fuse.Status {
return fuse.EACCES
}

View File

@ -22,6 +22,8 @@ import (
"os" "os"
"sort" "sort"
"camli/blobref"
"camli/client"
"camli/third_party/github.com/hanwen/go-fuse/fuse" "camli/third_party/github.com/hanwen/go-fuse/fuse"
) )
@ -45,23 +47,19 @@ func main() {
threaded := flag.Bool("threaded", true, "switch off threading; print debugging messages.") threaded := flag.Bool("threaded", true, "switch off threading; print debugging messages.")
flag.Parse() flag.Parse()
if flag.NArg() < 2 { if flag.NArg() < 2 {
// TODO - where to get program name? fmt.Println("usage: cammount <blobref> <mountpoint>")
fmt.Println("usage: main ORIGINAL MOUNTPOINT")
os.Exit(2) os.Exit(2)
} }
orig := flag.Arg(0) root := blobref.Parse(flag.Arg(0))
fs := fuse.NewLoopbackFileSystem(orig) if root == nil {
fmt.Printf("Error parsing root blobref: %q\n", root)
os.Exit(2)
}
client := client.NewOrFail() // automatic from flags
fs := NewCamliFileSystem(client, root)
timing := fuse.NewTimingPathFilesystem(fs) timing := fuse.NewTimingPathFilesystem(fs)
var opts fuse.PathFileSystemConnectorOptions
opts.AttrTimeout = 1.0
opts.EntryTimeout = 1.0
opts.NegativeTimeout = 1.0
fs.SetOptions(&opts)
conn := fuse.NewPathFileSystemConnector(timing) conn := fuse.NewPathFileSystemConnector(timing)
rawTiming := fuse.NewTimingRawFilesystem(conn) rawTiming := fuse.NewTimingRawFilesystem(conn)
@ -75,7 +73,7 @@ func main() {
os.Exit(1) os.Exit(1)
} }
fmt.Printf("Mounted %s on %s (threaded=%v, debug=%v)\n", orig, mountPoint, *threaded, *debug) fmt.Printf("Mounted %s on %s (threaded=%v, debug=%v)\n", root.String(), mountPoint, *threaded, *debug)
state.Loop(*threaded) state.Loop(*threaded)
fmt.Println("Finished", state.Stats()) fmt.Println("Finished", state.Stats())

View File

@ -55,6 +55,28 @@ type Superset struct {
Permanode string "permaNode" Permanode string "permaNode"
Attribute string "attribute" Attribute string "attribute"
Value string "value" Value string "value"
FileName string "fileName"
FileNameBytes []byte "fileNameBytes" // TODO: needs custom UnmarshalJSON?
UnixPermission string "unixPermission"
UnixOwnerId int "unixOwnerId"
UnixOwner string "unixOwner"
UnixGroupId int "unixGroupId"
UnixGroup string "unixGroup"
UnixMtime string "unixMtime"
UnixCtime string "unixCtime"
UnixAtime string "unixAtime"
Size int64 "size" // for files
Entries string "entries" // for directories, a blobref to a static-set
Members []string "members" // for static sets (for directory static-sets:
// blobrefs to child dirs/files)
}
func (ss *Superset) HasFilename(name string) bool {
// TODO: use filename bytes too
return ss.FileName == name
} }
var DefaultStatHasher = &defaultStatHasher{} var DefaultStatHasher = &defaultStatHasher{}