aboutsummaryrefslogblamecommitdiffstats
path: root/swarm/fuse/fuse_dir.go
blob: 7f66451f1035504dd8e09ef3e613bd9e9435c83e (plain) (tree)




















                                                                                  


                       


                           
                                                   
                                  






















                                                               
                                                     












                                                                   

                               







                                                                                                                      
                                                













                                                                            
                                       










                                                                                                                                 
                                                                          











                                                                                        
                                                                          



































                                                                                         

                                                                             




                                                       
 
// Copyright 2017 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.

// +build linux darwin freebsd

package fuse

import (
    "os"
    "path/filepath"
    "sync"

    "bazil.org/fuse"
    "bazil.org/fuse/fs"
    "github.com/ethereum/go-ethereum/swarm/log"
    "golang.org/x/net/context"
)

var (
    _ fs.Node                = (*SwarmDir)(nil)
    _ fs.NodeRequestLookuper = (*SwarmDir)(nil)
    _ fs.HandleReadDirAller  = (*SwarmDir)(nil)
    _ fs.NodeCreater         = (*SwarmDir)(nil)
    _ fs.NodeRemover         = (*SwarmDir)(nil)
    _ fs.NodeMkdirer         = (*SwarmDir)(nil)
)

type SwarmDir struct {
    inode       uint64
    name        string
    path        string
    directories []*SwarmDir
    files       []*SwarmFile

    mountInfo *MountInfo
    lock      *sync.RWMutex
}

func NewSwarmDir(fullpath string, minfo *MountInfo) *SwarmDir {
    log.Debug("swarmfs", "NewSwarmDir", fullpath)
    newdir := &SwarmDir{
        inode:       NewInode(),
        name:        filepath.Base(fullpath),
        path:        fullpath,
        directories: []*SwarmDir{},
        files:       []*SwarmFile{},
        mountInfo:   minfo,
        lock:        &sync.RWMutex{},
    }
    return newdir
}

func (sd *SwarmDir) Attr(ctx context.Context, a *fuse.Attr) error {
    sd.lock.RLock()
    defer sd.lock.RUnlock()
    a.Inode = sd.inode
    a.Mode = os.ModeDir | 0700
    a.Uid = uint32(os.Getuid())
    a.Gid = uint32(os.Getegid())
    return nil
}

func (sd *SwarmDir) Lookup(ctx context.Context, req *fuse.LookupRequest, resp *fuse.LookupResponse) (fs.Node, error) {
    log.Debug("swarmfs", "Lookup", req.Name)
    for _, n := range sd.files {
        if n.name == req.Name {
            return n, nil
        }
    }
    for _, n := range sd.directories {
        if n.name == req.Name {
            return n, nil
        }
    }
    return nil, fuse.ENOENT
}

func (sd *SwarmDir) ReadDirAll(ctx context.Context) ([]fuse.Dirent, error) {
    log.Debug("swarmfs ReadDirAll")
    var children []fuse.Dirent
    for _, file := range sd.files {
        children = append(children, fuse.Dirent{Inode: file.inode, Type: fuse.DT_File, Name: file.name})
    }
    for _, dir := range sd.directories {
        children = append(children, fuse.Dirent{Inode: dir.inode, Type: fuse.DT_Dir, Name: dir.name})
    }
    return children, nil
}

func (sd *SwarmDir) Create(ctx context.Context, req *fuse.CreateRequest, resp *fuse.CreateResponse) (fs.Node, fs.Handle, error) {
    log.Debug("swarmfs Create", "path", sd.path, "req.Name", req.Name)

    newFile := NewSwarmFile(sd.path, req.Name, sd.mountInfo)
    newFile.fileSize = 0 // 0 means, file is not in swarm yet and it is just created

    sd.lock.Lock()
    defer sd.lock.Unlock()
    sd.files = append(sd.files, newFile)

    return newFile, newFile, nil
}

func (sd *SwarmDir) Remove(ctx context.Context, req *fuse.RemoveRequest) error {
    log.Debug("swarmfs Remove", "path", sd.path, "req.Name", req.Name)

    if req.Dir && sd.directories != nil {
        newDirs := []*SwarmDir{}
        for _, dir := range sd.directories {
            if dir.name == req.Name {
                removeDirectoryFromSwarm(dir)
            } else {
                newDirs = append(newDirs, dir)
            }
        }
        if len(sd.directories) > len(newDirs) {
            sd.lock.Lock()
            defer sd.lock.Unlock()
            sd.directories = newDirs
        }
        return nil
    } else if !req.Dir && sd.files != nil {
        newFiles := []*SwarmFile{}
        for _, f := range sd.files {
            if f.name == req.Name {
                removeFileFromSwarm(f)
            } else {
                newFiles = append(newFiles, f)
            }
        }
        if len(sd.files) > len(newFiles) {
            sd.lock.Lock()
            defer sd.lock.Unlock()
            sd.files = newFiles
        }
        return nil
    }
    return fuse.ENOENT
}

func (sd *SwarmDir) Mkdir(ctx context.Context, req *fuse.MkdirRequest) (fs.Node, error) {
    log.Debug("swarmfs Mkdir", "path", sd.path, "req.Name", req.Name)
    newDir := NewSwarmDir(filepath.Join(sd.path, req.Name), sd.mountInfo)
    sd.lock.Lock()
    defer sd.lock.Unlock()
    sd.directories = append(sd.directories, newDir)

    return newDir, nil
}