readonly_unix.go 5.05 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

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
17
	"github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/goprotobuf/proto"
Jeromy's avatar
Jeromy committed
18

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"
24 25
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
26
var log = u.Logger("ipfs")
Jeromy's avatar
Jeromy committed
27

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

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

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

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

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

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
53
// Lookup performs a lookup under this node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
54
func (s *Root) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) {
Jeromy's avatar
Jeromy committed
55
	log.Debug("Root Lookup: '%s'", name)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56 57 58 59 60 61 62 63 64 65 66 67 68
	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
69 70
}

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

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
77
// Node is the core object representing a filesystem tree node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
78
type Node struct {
Jeromy's avatar
Jeromy committed
79 80 81 82 83 84 85 86 87
	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)
88 89
}

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

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

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

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
124
// ReadDir reads the link structure as directory entries
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
125
func (s *Node) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) {
Jeromy's avatar
Jeromy committed
126
	log.Debug("Node ReadDir")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
127 128 129 130 131 132 133 134 135 136 137 138 139
	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
140 141
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
142
// ReadAll reads the object data as file data
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
143
func (s *Node) ReadAll(intr fs.Intr) ([]byte, fuse.Error) {
144
	u.DOut("Read node.\n")
145
	r, err := mdag.NewDagReader(s.Nd, s.Ipfs.DAG)
146 147 148 149 150 151
	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)
152 153
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
154 155
// 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
156 157 158 159 160 161 162 163
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
164 165 166 167 168 169 170
		for {
			err := Unmount(fpath)
			if err == nil {
				return
			}
			time.Sleep(time.Millisecond * 10)
		}
171
		ipfs.Network.Close()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191
	}()

	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
192

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
193 194 195
// Unmount attempts to unmount the provided FUSE mount point, forcibly
// if necessary.
func Unmount(point string) error {
Jeromy's avatar
Jeromy committed
196
	log.Info("Unmounting %s...", point)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
197 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

	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
	}
223
}