readonly_unix.go 5.04 KB
Newer Older
1 2
// A Go mirror of libfuse's hello.c

3
// +build linux darwin freebsd
4

5 6 7
package readonly

import (
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
8
	"fmt"
9
	"io/ioutil"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10 11 12 13 14 15
	"os"
	"os/exec"
	"os/signal"
	"runtime"
	"syscall"
	"time"
16

Jeromy's avatar
Jeromy committed
17 18
	"code.google.com/p/goprotobuf/proto"

19 20
	"github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse"
	"github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse/fs"
21 22 23
	core "github.com/jbenet/go-ipfs/core"
	mdag "github.com/jbenet/go-ipfs/merkledag"
	u "github.com/jbenet/go-ipfs/util"
Jeromy's avatar
Jeromy committed
24
	"github.com/op/go-logging"
25 26
)

Jeromy's avatar
Jeromy committed
27 28
var log = logging.MustGetLogger("ipfs")

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
29
// FileSystem is the readonly Ipfs Fuse Filesystem.
30
type FileSystem struct {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
31
	Ipfs *core.IpfsNode
32 33
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
34
// NewFileSystem constructs new fs using given core.IpfsNode instance.
35
func NewFileSystem(ipfs *core.IpfsNode) *FileSystem {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
36
	return &FileSystem{Ipfs: ipfs}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
37 38
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
39
// Root constructs the Root of the filesystem, a Root object.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
40
func (f FileSystem) Root() (fs.Node, fuse.Error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
41
	return &Root{Ipfs: f.Ipfs}, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
42 43
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
44
// Root is the root object of the filesystem tree.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
45
type Root struct {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
46
	Ipfs *core.IpfsNode
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
47 48
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
49
// Attr returns file attributes.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
50 51
func (*Root) Attr() fuse.Attr {
	return fuse.Attr{Mode: os.ModeDir | 0111} // -rw+x
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
52 53
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
54
// Lookup performs a lookup under this node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
55
func (s *Root) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) {
Jeromy's avatar
Jeromy committed
56
	log.Debug("Root Lookup: '%s'", name)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
57 58 59 60 61 62 63 64 65 66 67 68 69
	switch name {
	case "mach_kernel", ".hidden", "._.":
		// Just quiet some log noise on OS X.
		return nil, fuse.ENOENT
	}

	nd, err := s.Ipfs.Resolver.ResolvePath(name)
	if err != nil {
		// todo: make this error more versatile.
		return nil, fuse.ENOENT
	}

	return &Node{Ipfs: s.Ipfs, Nd: nd}, nil
70 71
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
72
// ReadDir reads a particular directory. Disallowed for root.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
73
func (*Root) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) {
Jeromy's avatar
Jeromy committed
74
	log.Debug("Read Root.")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
75
	return nil, fuse.EPERM
76 77
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
78
// Node is the core object representing a filesystem tree node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
79
type Node struct {
Jeromy's avatar
Jeromy committed
80 81 82 83 84 85 86 87 88
	Ipfs   *core.IpfsNode
	Nd     *mdag.Node
	fd     *mdag.DagReader
	cached *mdag.PBData
}

func (s *Node) loadData() error {
	s.cached = new(mdag.PBData)
	return proto.Unmarshal(s.Nd.Data, s.cached)
89 90
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
91
// Attr returns the attributes of a given node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
92
func (s *Node) Attr() fuse.Attr {
Jeromy's avatar
Jeromy committed
93
	log.Debug("Node attr.")
Jeromy's avatar
Jeromy committed
94 95 96 97 98
	if s.cached == nil {
		s.loadData()
	}
	switch s.cached.GetType() {
	case mdag.PBData_Directory:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99
		return fuse.Attr{Mode: os.ModeDir | 0555}
Jeromy's avatar
Jeromy committed
100 101
	case mdag.PBData_File, mdag.PBData_Raw:
		size, _ := s.Nd.Size()
Jeromy's avatar
Jeromy committed
102 103 104 105 106
		return fuse.Attr{
			Mode:   0444,
			Size:   uint64(size),
			Blocks: uint64(len(s.Nd.Links)),
		}
Jeromy's avatar
Jeromy committed
107 108 109
	default:
		u.PErr("Invalid data type.")
		return fuse.Attr{}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
110
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
111
}
112

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
113
// Lookup performs a lookup under this node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
114
func (s *Node) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) {
Jeromy's avatar
Jeromy committed
115
	log.Debug("Lookup '%s'", name)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
116 117 118 119 120
	nd, err := s.Ipfs.Resolver.ResolveLinks(s.Nd, []string{name})
	if err != nil {
		// todo: make this error more versatile.
		return nil, fuse.ENOENT
	}
121

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
122
	return &Node{Ipfs: s.Ipfs, Nd: nd}, nil
123 124
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
125
// ReadDir reads the link structure as directory entries
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
126
func (s *Node) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) {
Jeromy's avatar
Jeromy committed
127
	log.Debug("Node ReadDir")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
128 129 130 131 132 133 134 135 136 137 138 139 140
	entries := make([]fuse.Dirent, len(s.Nd.Links))
	for i, link := range s.Nd.Links {
		n := link.Name
		if len(n) == 0 {
			n = link.Hash.B58String()
		}
		entries[i] = fuse.Dirent{Name: n, Type: fuse.DT_File}
	}

	if len(entries) > 0 {
		return entries, nil
	}
	return nil, fuse.ENOENT
141 142
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
143
// ReadAll reads the object data as file data
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
144
func (s *Node) ReadAll(intr fs.Intr) ([]byte, fuse.Error) {
145
	u.DOut("Read node.\n")
146
	r, err := mdag.NewDagReader(s.Nd, s.Ipfs.DAG)
147 148 149 150 151 152
	if err != nil {
		return nil, err
	}
	// this is a terrible function... 'ReadAll'?
	// what if i have a 6TB file? GG RAM.
	return ioutil.ReadAll(r)
153 154
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
155 156
// Mount mounts an IpfsNode instance at a particular path. It
// serves until the process receives exit signals (to Unmount).
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
157 158 159 160 161 162 163 164
func Mount(ipfs *core.IpfsNode, fpath string) error {

	sigc := make(chan os.Signal, 1)
	signal.Notify(sigc, syscall.SIGHUP, syscall.SIGINT,
		syscall.SIGTERM, syscall.SIGQUIT)

	go func() {
		<-sigc
165 166 167 168 169 170 171
		for {
			err := Unmount(fpath)
			if err == nil {
				return
			}
			time.Sleep(time.Millisecond * 10)
		}
172
		ipfs.Network.Close()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
	}()

	c, err := fuse.Mount(fpath)
	if err != nil {
		return err
	}
	defer c.Close()

	err = fs.Serve(c, FileSystem{Ipfs: ipfs})
	if err != nil {
		return err
	}

	// check if the mount process has an error to report
	<-c.Ready
	if err := c.MountError; err != nil {
		return err
	}
	return nil
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
193

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
194 195 196
// Unmount attempts to unmount the provided FUSE mount point, forcibly
// if necessary.
func Unmount(point string) error {
Jeromy's avatar
Jeromy committed
197
	log.Info("Unmounting %s...", point)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223

	var cmd *exec.Cmd
	switch runtime.GOOS {
	case "darwin":
		cmd = exec.Command("diskutil", "umount", "force", point)
	case "linux":
		cmd = exec.Command("fusermount", "-u", point)
	default:
		return fmt.Errorf("unmount: unimplemented")
	}

	errc := make(chan error, 1)
	go func() {
		if err := exec.Command("umount", point).Run(); err == nil {
			errc <- err
		}
		// retry to unmount with the fallback cmd
		errc <- cmd.Run()
	}()

	select {
	case <-time.After(1 * time.Second):
		return fmt.Errorf("umount timeout")
	case err := <-errc:
		return err
	}
224
}